home *** CD-ROM | disk | FTP | other *** search
/ SGI Freeware 1999 August / SGI Freeware 1999 August.iso / dist / samba.idb / usr / samba / src / source / loadparm.c.z / loadparm.c
Encoding:
C/C++ Source or Header  |  1998-10-28  |  84.9 KB  |  2,503 lines

  1. /* 
  2.    Unix SMB/Netbios implementation.
  3.    Version 1.9.
  4.    Parameter loading functions
  5.    Copyright (C) Karl Auer 1993-1998
  6.  
  7.    Largely re-written by Andrew Tridgell, September 1994
  8.    
  9.    This program is free software; you can redistribute it and/or modify
  10.    it under the terms of the GNU General Public License as published by
  11.    the Free Software Foundation; either version 2 of the License, or
  12.    (at your option) any later version.
  13.    
  14.    This program is distributed in the hope that it will be useful,
  15.    but WITHOUT ANY WARRANTY; without even the implied warranty of
  16.    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  17.    GNU General Public License for more details.
  18.    
  19.    You should have received a copy of the GNU General Public License
  20.    along with this program; if not, write to the Free Software
  21.    Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
  22. */
  23.  
  24. /*
  25.  *  Load parameters.
  26.  *
  27.  *  This module provides suitable callback functions for the params
  28.  *  module. It builds the internal table of service details which is
  29.  *  then used by the rest of the server.
  30.  *
  31.  * To add a parameter:
  32.  *
  33.  * 1) add it to the global or service structure definition
  34.  * 2) add it to the parm_table
  35.  * 3) add it to the list of available functions (eg: using FN_GLOBAL_STRING())
  36.  * 4) If it's a global then initialise it in init_globals. If a local
  37.  *    (ie. service) parameter then initialise it in the sDefault structure
  38.  *  
  39.  *
  40.  * Notes:
  41.  *   The configuration file is processed sequentially for speed. It is NOT
  42.  *   accessed randomly as happens in 'real' Windows. For this reason, there
  43.  *   is a fair bit of sequence-dependent code here - ie., code which assumes
  44.  *   that certain things happen before others. In particular, the code which
  45.  *   happens at the boundary between sections is delicately poised, so be
  46.  *   careful!
  47.  *
  48.  */
  49.  
  50. #include "includes.h"
  51.  
  52. /* Set default coding system for KANJI if none specified in Makefile. */
  53. /* 
  54.  * We treat KANJI specially due to historical precedent (it was the
  55.  * first non-english codepage added to Samba). With the new dynamic
  56.  * codepage support this is not needed anymore.
  57.  *
  58.  * The define 'KANJI' is being overloaded to mean 'use kanji codepage
  59.  * by default' and also 'this is the filename-to-disk conversion 
  60.  * method to use'. This really should be removed and all control
  61.  * over this left in the smb.conf parameters 'client codepage'
  62.  * and 'coding system'.
  63.  */
  64. #ifndef KANJI
  65. #define KANJI "sbcs"
  66. #endif /* KANJI */
  67.  
  68. BOOL bLoaded = False;
  69.  
  70. extern int DEBUGLEVEL;
  71. extern pstring user_socket_options;
  72. extern pstring myname;
  73.  
  74. #ifndef GLOBAL_NAME
  75. #define GLOBAL_NAME "global"
  76. #endif
  77.  
  78. #ifndef PRINTERS_NAME
  79. #define PRINTERS_NAME "printers"
  80. #endif
  81.  
  82. #ifndef HOMES_NAME
  83. #define HOMES_NAME "homes"
  84. #endif
  85.  
  86. /* some helpful bits */
  87. #define pSERVICE(i) ServicePtrs[i]
  88. #define iSERVICE(i) (*pSERVICE(i))
  89. #define LP_SNUM_OK(iService) (((iService) >= 0) && ((iService) < iNumServices) && iSERVICE(iService).valid)
  90. #define VALID(i) iSERVICE(i).valid
  91.  
  92. /* these are the types of parameter we have */
  93. typedef enum
  94. {
  95.   P_BOOL,P_BOOLREV,P_CHAR,P_INTEGER,P_OCTAL,
  96.   P_STRING,P_USTRING,P_GSTRING,P_UGSTRING,P_ENUM
  97. } parm_type;
  98.  
  99. typedef enum
  100. {
  101.   P_LOCAL,P_GLOBAL,P_NONE
  102. } parm_class;
  103.  
  104. int keepalive=0;
  105. extern BOOL use_getwd_cache;
  106.  
  107. extern int extra_time_offset;
  108.  
  109. /* 
  110.  * This structure describes global (ie., server-wide) parameters.
  111.  */
  112. typedef struct
  113. {
  114.   char *szPrintcapname;
  115.   char *szLockDir;
  116.   char *szRootdir;
  117.   char *szDefaultService;
  118.   char *szDfree;
  119.   char *szMsgCommand;
  120.   char *szHostsEquiv;
  121.   char *szServerString;
  122.   char *szAutoServices;
  123.   char *szPasswdProgram;
  124.   char *szPasswdChat;
  125.   char *szLogFile;
  126.   char *szConfigFile;
  127.   char *szSMBPasswdFile;
  128.   char *szPasswordServer;
  129.   char *szSocketOptions;
  130.   char *szValidChars;
  131.   char *szWorkGroup;
  132.   char *szDomainAdminUsers;
  133.   char *szDomainGuestUsers;
  134.   char *szDomainHostsallow; 
  135.   char *szDomainHostsdeny;
  136.   char *szUsernameMap;
  137.   char *szCharacterSet;
  138.   char *szLogonScript;
  139.   char *szLogonPath;
  140.   char *szLogonDrive;
  141.   char *szLogonHome;
  142.   char *szSmbrun;
  143.   char *szWINSserver;
  144.   char *szCodingSystem;
  145.   char *szInterfaces;
  146.   char *szRemoteAnnounce;
  147.   char *szRemoteBrowseSync;
  148.   char *szSocketAddress;
  149.   char *szNISHomeMapName;
  150.   char *szAnnounceVersion; /* This is initialised in init_globals */
  151.   char *szNetbiosAliases;
  152.   char *szDomainSID;
  153.   char *szDomainOtherSIDs;
  154.   char *szDomainGroups;
  155.   char *szDriverFile;
  156.   char *szNameResolveOrder;
  157.   int max_log_size;
  158.   int mangled_stack;
  159.   int max_xmit;
  160.   int max_mux;
  161.   int max_packet;
  162.   int pwordlevel;
  163.   int unamelevel;
  164.   int deadtime;
  165.   int maxprotocol;
  166.   int security;
  167.   int maxdisksize;
  168.   int lpqcachetime;
  169.   int syslog;
  170.   int os_level;
  171.   int max_ttl;
  172.   int max_wins_ttl;
  173.   int min_wins_ttl;
  174.   int ReadSize;
  175.   int lm_announce;
  176.   int lm_interval;
  177.   int shmem_size;
  178.   int client_code_page;
  179.   int announce_as;   /* This is initialised in init_globals */
  180.   BOOL bDNSproxy;
  181.   BOOL bWINSsupport;
  182.   BOOL bWINSproxy;
  183.   BOOL bLocalMaster;
  184.   BOOL bPreferredMaster;
  185.   BOOL bDomainController;
  186.   BOOL bDomainMaster;
  187.   BOOL bDomainLogons;
  188.   BOOL bEncryptPasswords;
  189.   BOOL bUpdateEncrypt;
  190.   BOOL bStripDot;
  191.   BOOL bNullPasswords;
  192.   BOOL bLoadPrinters;
  193.   BOOL bUseRhosts;
  194.   BOOL bReadRaw;
  195.   BOOL bWriteRaw;
  196.   BOOL bReadPrediction;
  197.   BOOL bReadbmpx;
  198.   BOOL bSyslogOnly;
  199.   BOOL bBrowseList;
  200.   BOOL bUnixRealname;
  201.   BOOL bNISHomeMap;
  202.   BOOL bTimeServer;
  203.   BOOL bBindInterfacesOnly;
  204.   BOOL bNetWkstaUserLogon;
  205.   BOOL bUnixPasswdSync;
  206.   BOOL bPasswdChatDebug;
  207.   BOOL bOleLockingCompat;
  208. } global;
  209.  
  210. static global Globals;
  211.  
  212.  
  213.  
  214. /* 
  215.  * This structure describes a single service. 
  216.  */
  217. typedef struct
  218. {
  219.   BOOL valid;
  220.   char *szService;
  221.   char *szPath;
  222.   char *szUsername;
  223.   char *szGuestaccount;
  224.   char *szInvalidUsers;
  225.   char *szValidUsers;
  226.   char *szAdminUsers;
  227.   char *szCopy;
  228.   char *szInclude;
  229.   char *szPreExec;
  230.   char *szPostExec;
  231.   char *szRootPreExec;
  232.   char *szRootPostExec;
  233.   char *szPrintcommand;
  234.   char *szLpqcommand;
  235.   char *szLprmcommand;
  236.   char *szLppausecommand;
  237.   char *szLpresumecommand;
  238.   char *szQueuepausecommand;
  239.   char *szQueueresumecommand;
  240.   char *szPrintername;
  241.   char *szPrinterDriver;
  242.   char *szPrinterDriverLocation;
  243.   char *szDontdescend;
  244.   char *szHostsallow;
  245.   char *szHostsdeny;
  246.   char *szMagicScript;
  247.   char *szMagicOutput;
  248.   char *szMangledMap;
  249.   char *szVetoFiles;
  250.   char *szHideFiles;
  251.   char *szVetoOplockFiles;
  252.   char *comment;
  253.   char *force_user;
  254.   char *force_group;
  255.   char *readlist;
  256.   char *writelist;
  257.   char *volume;
  258.   int  iMinPrintSpace;
  259.   int  iCreate_mask;
  260.   int  iCreate_force_mode;
  261.   int  iDir_mask;
  262.   int  iDir_force_mode;
  263.   int  iMaxConnections;
  264.   int  iDefaultCase;
  265.   int  iPrinting;
  266.   BOOL bAlternatePerm;
  267.   BOOL bRevalidate;
  268.   BOOL bCaseSensitive;
  269.   BOOL bCasePreserve;
  270.   BOOL bShortCasePreserve;
  271.   BOOL bCaseMangle;
  272.   BOOL status;
  273.   BOOL bHideDotFiles;
  274.   BOOL bBrowseable;
  275.   BOOL bAvailable;
  276.   BOOL bRead_only;
  277.   BOOL bNo_set_dir;
  278.   BOOL bGuest_only;
  279.   BOOL bGuest_ok;
  280.   BOOL bPrint_ok;
  281.   BOOL bPostscript;
  282.   BOOL bMap_system;
  283.   BOOL bMap_hidden;
  284.   BOOL bMap_archive;
  285.   BOOL bLocking;
  286.   BOOL bStrictLocking;
  287.   BOOL bShareModes;
  288.   BOOL bOpLocks;
  289.   BOOL bOnlyUser;
  290.   BOOL bMangledNames;
  291.   BOOL bWidelinks;
  292.   BOOL bSymlinks;
  293.   BOOL bSyncAlways;
  294.   BOOL bStrictSync;
  295.   char magic_char;
  296.   BOOL *copymap;
  297.   BOOL bDeleteReadonly;
  298.   BOOL bFakeOplocks;
  299.   BOOL bDeleteVetoFiles;
  300.   BOOL bDosFiletimes;
  301.   BOOL bDosFiletimeResolution;
  302.   BOOL bFakeDirCreateTimes;
  303.   char dummy[3]; /* for alignment */
  304. } service;
  305.  
  306.  
  307. /* This is a default service used to prime a services structure */
  308. static service sDefault = 
  309. {
  310.   True,   /* valid */
  311.   NULL,    /* szService */
  312.   NULL,    /* szPath */
  313.   NULL,    /* szUsername */
  314.   NULL,    /* szGuestAccount  - this is set in init_globals() */
  315.   NULL,    /* szInvalidUsers */
  316.   NULL,    /* szValidUsers */
  317.   NULL,    /* szAdminUsers */
  318.   NULL,    /* szCopy */
  319.   NULL,    /* szInclude */
  320.   NULL,    /* szPreExec */
  321.   NULL,    /* szPostExec */
  322.   NULL,    /* szRootPreExec */
  323.   NULL,    /* szRootPostExec */
  324.   NULL,    /* szPrintcommand */
  325.   NULL,    /* szLpqcommand */
  326.   NULL,    /* szLprmcommand */
  327.   NULL,    /* szLppausecommand */
  328.   NULL,    /* szLpresumecommand */
  329.   NULL,    /* szQueuepausecommand */
  330.   NULL,    /* szQueueresumecommand */
  331.   NULL,    /* szPrintername */
  332.   NULL,    /* szPrinterDriver - this is set in init_globals() */
  333.   NULL,    /* szPrinterDriverLocation */
  334.   NULL,    /* szDontdescend */
  335.   NULL,    /* szHostsallow */
  336.   NULL,    /* szHostsdeny */
  337.   NULL,    /* szMagicScript */
  338.   NULL,    /* szMagicOutput */
  339.   NULL,    /* szMangledMap */
  340.   NULL,    /* szVetoFiles */
  341.   NULL,    /* szHideFiles */
  342.   NULL,    /* szVetoOplockFiles */
  343.   NULL,    /* comment */
  344.   NULL,    /* force user */
  345.   NULL,    /* force group */
  346.   NULL,    /* readlist */
  347.   NULL,    /* writelist */
  348.   NULL,    /* volume */
  349.   0,       /* iMinPrintSpace */
  350.   0744,    /* iCreate_mask */
  351.   0000,    /* iCreate_force_mode */
  352.   0755,    /* iDir_mask */
  353.   0000,    /* iDir_force_mode */
  354.   0,       /* iMaxConnections */
  355.   CASE_LOWER, /* iDefaultCase */
  356.   DEFAULT_PRINTING, /* iPrinting */
  357.   False,   /* bAlternatePerm */
  358.   False,   /* revalidate */
  359.   False,   /* case sensitive */
  360.   False,   /* case preserve */
  361.   False,   /* short case preserve */
  362.   False,  /* case mangle */
  363.   True,  /* status */
  364.   True,  /* bHideDotFiles */
  365.   True,  /* bBrowseable */
  366.   True,  /* bAvailable */
  367.   True,  /* bRead_only */
  368.   True,  /* bNo_set_dir */
  369.   False, /* bGuest_only */
  370.   False, /* bGuest_ok */
  371.   False, /* bPrint_ok */
  372.   False, /* bPostscript */
  373.   False, /* bMap_system */
  374.   False, /* bMap_hidden */
  375.   True,  /* bMap_archive */
  376.   True,  /* bLocking */
  377.   False,  /* bStrictLocking */
  378.   True,  /* bShareModes */
  379.   True,  /* bOpLocks */
  380.   False, /* bOnlyUser */
  381.   True,  /* bMangledNames */
  382.   True,  /* bWidelinks */
  383.   True,  /* bSymlinks */
  384.   False, /* bSyncAlways */
  385.   False, /* bStrictSync */
  386.   '~',   /* magic char */
  387.   NULL,  /* copymap */
  388.   False, /* bDeleteReadonly */
  389.   False, /* bFakeOplocks */
  390.   False, /* bDeleteVetoFiles */
  391.   False, /* bDosFiletimes */
  392.   False, /* bDosFiletimeResolution */
  393.   False, /* bFakeDirCreateTimes */
  394.   ""     /* dummy */
  395. };
  396.  
  397.  
  398.  
  399. /* local variables */
  400. static service **ServicePtrs = NULL;
  401. static int iNumServices = 0;
  402. static int iServiceIndex = 0;
  403. static BOOL bInGlobalSection = True;
  404. static BOOL bGlobalOnly = False;
  405. static int default_server_announce;
  406.  
  407. #define NUMPARAMETERS (sizeof(parm_table) / sizeof(struct parm_struct))
  408.  
  409. /* prototypes for the special type handlers */
  410. static BOOL handle_valid_chars(char *pszParmValue, char **ptr);
  411. static BOOL handle_include(char *pszParmValue, char **ptr);
  412. static BOOL handle_copy(char *pszParmValue, char **ptr);
  413. static BOOL handle_character_set(char *pszParmValue,char **ptr);
  414. static BOOL handle_coding_system(char *pszParmValue,char **ptr);
  415.  
  416. static void set_default_server_announce_type(void);
  417.  
  418. struct enum_list {
  419.     int value;
  420.     char *name;
  421. };
  422.  
  423. static struct enum_list enum_protocol[] = {{PROTOCOL_NT1, "NT1"}, {PROTOCOL_LANMAN2, "LANMAN2"}, 
  424.                        {PROTOCOL_LANMAN1, "LANMAN1"}, {PROTOCOL_CORE,"CORE"}, 
  425.                        {PROTOCOL_COREPLUS, "COREPLUS"}, 
  426.                        {PROTOCOL_COREPLUS, "CORE+"}, {-1, NULL}};
  427.  
  428. static struct enum_list enum_security[] = {{SEC_SHARE, "SHARE"},  {SEC_USER, "USER"}, 
  429.                        {SEC_SERVER, "SERVER"}, {-1, NULL}};
  430.  
  431. static struct enum_list enum_printing[] = {{PRINT_SYSV, "sysv"}, {PRINT_AIX, "aix"}, 
  432.                        {PRINT_HPUX, "hpux"}, {PRINT_BSD, "bsd"},
  433.                        {PRINT_QNX, "qnx"},   {PRINT_PLP, "plp"},
  434.                        {PRINT_LPRNG, "lprng"}, {PRINT_SOFTQ, "softq"},
  435.                        {-1, NULL}};
  436.  
  437. static struct enum_list enum_announce_as[] = {{ANNOUNCE_AS_NT, "NT"}, {ANNOUNCE_AS_WIN95, "win95"},
  438.                           {ANNOUNCE_AS_WFW, "WfW"}, {-1, NULL}};
  439.  
  440. static struct enum_list enum_case[] = {{CASE_LOWER, "lower"}, {CASE_UPPER, "upper"}, {-1, NULL}};
  441.  
  442. static struct enum_list enum_lm_announce[] = {{0, "False"}, {1, "True"}, {2, "Auto"}, {-1, NULL}};
  443.  
  444. static struct parm_struct
  445. {
  446.     char *label;
  447.     parm_type type;
  448.     parm_class class;
  449.     void *ptr;
  450.     BOOL (*special)(char *, char **);
  451.     struct enum_list *enum_list;
  452. } parm_table[] =
  453. {
  454.   {"debuglevel",       P_INTEGER, P_GLOBAL, &DEBUGLEVEL,                NULL,   NULL},
  455.   {"log level",        P_INTEGER, P_GLOBAL, &DEBUGLEVEL,                NULL,   NULL},
  456.   {"syslog",           P_INTEGER, P_GLOBAL, &Globals.syslog,            NULL,   NULL},
  457.   {"syslog only",      P_BOOL,    P_GLOBAL, &Globals.bSyslogOnly,       NULL,   NULL},
  458.   {"protocol",         P_ENUM,    P_GLOBAL, &Globals.maxprotocol,       NULL,   enum_protocol},
  459.   {"security",         P_ENUM,    P_GLOBAL, &Globals.security,          NULL,   enum_security},
  460.   {"max disk size",    P_INTEGER, P_GLOBAL, &Globals.maxdisksize,       NULL,   NULL},
  461.   {"lpq cache time",   P_INTEGER, P_GLOBAL, &Globals.lpqcachetime,      NULL,   NULL},
  462.   {"announce as",      P_ENUM,    P_GLOBAL, &Globals.announce_as,       NULL,   enum_announce_as},
  463.   {"encrypt passwords",P_BOOL,    P_GLOBAL, &Globals.bEncryptPasswords, NULL,   NULL},
  464.   {"update encrypted", P_BOOL,    P_GLOBAL, &Globals.bUpdateEncrypt,    NULL,   NULL},
  465.   {"getwd cache",      P_BOOL,    P_GLOBAL, &use_getwd_cache,           NULL,   NULL},
  466.   {"read prediction",  P_BOOL,    P_GLOBAL, &Globals.bReadPrediction,   NULL,   NULL},
  467.   {"read bmpx",        P_BOOL,    P_GLOBAL, &Globals.bReadbmpx,         NULL,   NULL},
  468.   {"read raw",         P_BOOL,    P_GLOBAL, &Globals.bReadRaw,          NULL,   NULL},
  469.   {"write raw",        P_BOOL,    P_GLOBAL, &Globals.bWriteRaw,         NULL,   NULL},
  470.   {"use rhosts",       P_BOOL,    P_GLOBAL, &Globals.bUseRhosts,        NULL,   NULL},
  471.   {"load printers",    P_BOOL,    P_GLOBAL, &Globals.bLoadPrinters,     NULL,   NULL},
  472.   {"null passwords",   P_BOOL,    P_GLOBAL, &Globals.bNullPasswords,    NULL,   NULL},
  473.   {"strip dot",        P_BOOL,    P_GLOBAL, &Globals.bStripDot,         NULL,   NULL},
  474.   {"interfaces",       P_STRING,  P_GLOBAL, &Globals.szInterfaces,      NULL,   NULL},
  475.   {"bind interfaces only", P_BOOL,P_GLOBAL, &Globals.bBindInterfacesOnly,NULL,   NULL},
  476.   {"networkstation user login", P_BOOL,P_GLOBAL, &Globals.bNetWkstaUserLogon,NULL,   NULL},
  477.   {"password server",  P_STRING,  P_GLOBAL, &Globals.szPasswordServer,  NULL,   NULL},
  478.   {"socket options",   P_GSTRING, P_GLOBAL, user_socket_options,        NULL,   NULL},
  479.   {"netbios name",     P_UGSTRING,P_GLOBAL, myname,                     NULL,   NULL},
  480.   {"netbios aliases",  P_STRING,  P_GLOBAL, &Globals.szNetbiosAliases,  NULL,   NULL},
  481.   {"smbrun",           P_STRING,  P_GLOBAL, &Globals.szSmbrun,          NULL,   NULL},
  482.   {"log file",         P_STRING,  P_GLOBAL, &Globals.szLogFile,         NULL,   NULL},
  483.   {"config file",      P_STRING,  P_GLOBAL, &Globals.szConfigFile,      NULL,   NULL},
  484.   {"smb passwd file",  P_STRING,  P_GLOBAL, &Globals.szSMBPasswdFile,   NULL,   NULL},
  485.   {"hosts equiv",      P_STRING,  P_GLOBAL, &Globals.szHostsEquiv,      NULL,   NULL},
  486.   {"preload",          P_STRING,  P_GLOBAL, &Globals.szAutoServices,    NULL,   NULL},
  487.   {"auto services",    P_STRING,  P_GLOBAL, &Globals.szAutoServices,    NULL,   NULL},
  488.   {"server string",    P_STRING,  P_GLOBAL, &Globals.szServerString,    NULL,   NULL},
  489.   {"printcap name",    P_STRING,  P_GLOBAL, &Globals.szPrintcapname,    NULL,   NULL},
  490.   {"printcap",         P_STRING,  P_GLOBAL, &Globals.szPrintcapname,    NULL,   NULL},
  491.   {"lock dir",         P_STRING,  P_GLOBAL, &Globals.szLockDir,         NULL,   NULL},
  492.   {"lock directory",   P_STRING,  P_GLOBAL, &Globals.szLockDir,         NULL,   NULL},
  493.   {"root directory",   P_STRING,  P_GLOBAL, &Globals.szRootdir,         NULL,   NULL},
  494.   {"root dir",         P_STRING,  P_GLOBAL, &Globals.szRootdir,         NULL,   NULL},
  495.   {"root",             P_STRING,  P_GLOBAL, &Globals.szRootdir,         NULL,   NULL},
  496.   {"default service",  P_STRING,  P_GLOBAL, &Globals.szDefaultService,  NULL,   NULL},
  497.   {"default",          P_STRING,  P_GLOBAL, &Globals.szDefaultService,  NULL,   NULL},
  498.   {"message command",  P_STRING,  P_GLOBAL, &Globals.szMsgCommand,      NULL,   NULL},
  499.   {"dfree command",    P_STRING,  P_GLOBAL, &Globals.szDfree,           NULL,   NULL},
  500.   {"passwd program",   P_STRING,  P_GLOBAL, &Globals.szPasswdProgram,   NULL,   NULL},
  501.   {"passwd chat",      P_STRING,  P_GLOBAL, &Globals.szPasswdChat,      NULL,   NULL},
  502.   {"passwd chat debug",P_BOOL,    P_GLOBAL, &Globals.bPasswdChatDebug, NULL,   NULL},
  503.   {"valid chars",      P_STRING,  P_GLOBAL, &Globals.szValidChars,      handle_valid_chars, NULL},
  504.   {"workgroup",        P_USTRING, P_GLOBAL, &Globals.szWorkGroup,       NULL,   NULL},
  505. #ifdef NTDOMAIN
  506.   {"domain sid",       P_USTRING, P_GLOBAL, &Globals.szDomainSID,       NULL,   NULL},
  507.   {"domain other sids",P_STRING,  P_GLOBAL, &Globals.szDomainOtherSIDs, NULL,   NULL},
  508.   {"domain groups",    P_STRING,  P_GLOBAL, &Globals.szDomainGroups,    NULL,   NULL},
  509.   {"domain controller",P_BOOL  ,  P_GLOBAL, &Globals.bDomainController,NULL,   NULL},
  510.   {"domain admin users",P_STRING, P_GLOBAL, &Globals.szDomainAdminUsers, NULL,   NULL},
  511.   {"domain guest users",P_STRING, P_GLOBAL, &Globals.szDomainGuestUsers, NULL,   NULL},
  512.   {"domain hosts allow",P_STRING, P_GLOBAL, &Globals.szDomainHostsallow, NULL,   NULL},
  513.   {"domain allow hosts",P_STRING, P_GLOBAL, &Globals.szDomainHostsallow, NULL,   NULL},
  514.   {"domain hosts deny", P_STRING, P_GLOBAL, &Globals.szDomainHostsdeny,  NULL,   NULL},
  515.   {"domain deny hosts", P_STRING, P_GLOBAL, &Globals.szDomainHostsdeny,  NULL,   NULL},
  516. #endif /* NTDOMAIN */
  517.   {"username map",     P_STRING,  P_GLOBAL, &Globals.szUsernameMap,     NULL,   NULL},
  518.   {"character set",    P_STRING,  P_GLOBAL, &Globals.szCharacterSet,    handle_character_set, NULL},
  519.   {"logon script",     P_STRING,  P_GLOBAL, &Globals.szLogonScript,     NULL,   NULL},
  520.   {"logon path",       P_STRING,  P_GLOBAL, &Globals.szLogonPath,       NULL,   NULL},
  521.   {"logon drive",      P_STRING,  P_GLOBAL, &Globals.szLogonDrive,      NULL,   NULL},
  522.   {"logon home",       P_STRING,  P_GLOBAL, &Globals.szLogonHome,       NULL,   NULL},
  523.   {"remote announce",  P_STRING,  P_GLOBAL, &Globals.szRemoteAnnounce,  NULL,   NULL},
  524.   {"remote browse sync",P_STRING, P_GLOBAL, &Globals.szRemoteBrowseSync,NULL,   NULL},
  525.   {"socket address",   P_STRING,  P_GLOBAL, &Globals.szSocketAddress,   NULL,   NULL},
  526.   {"homedir map",      P_STRING,  P_GLOBAL, &Globals.szNISHomeMapName,  NULL,   NULL},
  527.   {"announce version", P_STRING,  P_GLOBAL, &Globals.szAnnounceVersion, NULL,   NULL},
  528.   {"max log size",     P_INTEGER, P_GLOBAL, &Globals.max_log_size,      NULL,   NULL},
  529.   {"mangled stack",    P_INTEGER, P_GLOBAL, &Globals.mangled_stack,     NULL,   NULL},
  530.   {"max mux",          P_INTEGER, P_GLOBAL, &Globals.max_mux,           NULL,   NULL},
  531.   {"max xmit",         P_INTEGER, P_GLOBAL, &Globals.max_xmit,          NULL,   NULL},
  532.   {"max packet",       P_INTEGER, P_GLOBAL, &Globals.max_packet,        NULL,   NULL},
  533.   {"name resolve order",  P_STRING,  P_GLOBAL, &Globals.szNameResolveOrder,  NULL,   NULL},
  534.   {"packet size",      P_INTEGER, P_GLOBAL, &Globals.max_packet,        NULL,   NULL},
  535.   {"password level",   P_INTEGER, P_GLOBAL, &Globals.pwordlevel,        NULL,   NULL},
  536.   {"username level",   P_INTEGER, P_GLOBAL, &Globals.unamelevel,        NULL,   NULL},
  537.   {"keepalive",        P_INTEGER, P_GLOBAL, &keepalive,                 NULL,   NULL},
  538.   {"deadtime",         P_INTEGER, P_GLOBAL, &Globals.deadtime,          NULL,   NULL},
  539.   {"time offset",      P_INTEGER, P_GLOBAL, &extra_time_offset,         NULL,   NULL},
  540.   {"read size",        P_INTEGER, P_GLOBAL, &Globals.ReadSize,          NULL,   NULL},
  541.   {"shared mem size",  P_INTEGER, P_GLOBAL, &Globals.shmem_size,        NULL,   NULL},
  542.   {"coding system",    P_STRING,  P_GLOBAL, &Globals.szCodingSystem,    handle_coding_system, NULL},
  543.   {"client code page", P_INTEGER, P_GLOBAL, &Globals.client_code_page,    NULL,   NULL},
  544.   {"os level",         P_INTEGER, P_GLOBAL, &Globals.os_level,          NULL,   NULL},
  545.   {"max ttl",          P_INTEGER, P_GLOBAL, &Globals.max_ttl,           NULL,   NULL},
  546.   {"max wins ttl",     P_INTEGER, P_GLOBAL, &Globals.max_wins_ttl,      NULL,   NULL},
  547.   {"min wins ttl",     P_INTEGER, P_GLOBAL, &Globals.min_wins_ttl,      NULL,   NULL},
  548.   {"lm announce",      P_ENUM,    P_GLOBAL, &Globals.lm_announce,       NULL,   enum_lm_announce},
  549.   {"lm interval",      P_INTEGER, P_GLOBAL, &Globals.lm_interval,       NULL,   NULL},
  550.   {"dns proxy",        P_BOOL,    P_GLOBAL, &Globals.bDNSproxy,         NULL,   NULL},
  551.   {"wins support",     P_BOOL,    P_GLOBAL, &Globals.bWINSsupport,      NULL,   NULL},
  552.   {"wins proxy",       P_BOOL,    P_GLOBAL, &Globals.bWINSproxy,        NULL,   NULL},
  553.   {"wins server",      P_STRING,  P_GLOBAL, &Globals.szWINSserver,      NULL,   NULL},
  554.   {"preferred master", P_BOOL,    P_GLOBAL, &Globals.bPreferredMaster,  NULL,   NULL},
  555.   {"prefered master",  P_BOOL,    P_GLOBAL, &Globals.bPreferredMaster,  NULL,   NULL},
  556.   {"local master",     P_BOOL,    P_GLOBAL, &Globals.bLocalMaster,      NULL,   NULL},
  557.   {"domain master",    P_BOOL,    P_GLOBAL, &Globals.bDomainMaster,     NULL,   NULL},
  558.   {"domain logons",    P_BOOL,    P_GLOBAL, &Globals.bDomainLogons,     NULL,   NULL},
  559.   {"browse list",      P_BOOL,    P_GLOBAL, &Globals.bBrowseList,       NULL,   NULL},
  560.   {"unix realname",    P_BOOL,    P_GLOBAL, &Globals.bUnixRealname,     NULL,   NULL},
  561.   {"NIS homedir",      P_BOOL,    P_GLOBAL, &Globals.bNISHomeMap,       NULL,   NULL},
  562.   {"unix password sync", P_BOOL,  P_GLOBAL, &Globals.bUnixPasswdSync,   NULL,   NULL},
  563.   {"time server",      P_BOOL,    P_GLOBAL, &Globals.bTimeServer,    NULL,   NULL},
  564.   {"ole locking compatibility",      P_BOOL,    P_GLOBAL, &Globals.bOleLockingCompat,    NULL,   NULL},
  565.   {"printer driver file", P_STRING,  P_GLOBAL, &Globals.szDriverFile,   NULL,   NULL},
  566.   {"-valid",           P_BOOL,    P_LOCAL,  &sDefault.valid,            NULL,   NULL},
  567.   {"comment",          P_STRING,  P_LOCAL,  &sDefault.comment,          NULL,   NULL},
  568.   {"copy",             P_STRING,  P_LOCAL,  &sDefault.szCopy,           handle_copy, NULL},
  569.   {"include",          P_STRING,  P_LOCAL,  &sDefault.szInclude,        handle_include, NULL},
  570.   {"exec",             P_STRING,  P_LOCAL,  &sDefault.szPreExec,        NULL,   NULL},
  571.   {"preexec",          P_STRING,  P_LOCAL,  &sDefault.szPreExec,        NULL,   NULL},
  572.   {"postexec",         P_STRING,  P_LOCAL,  &sDefault.szPostExec,       NULL,   NULL},
  573.   {"root preexec",     P_STRING,  P_LOCAL,  &sDefault.szRootPreExec,    NULL,   NULL},
  574.   {"root postexec",    P_STRING,  P_LOCAL,  &sDefault.szRootPostExec,   NULL,   NULL},
  575.   {"alternate permissions",P_BOOL,P_LOCAL,  &sDefault.bAlternatePerm,   NULL,   NULL},
  576.   {"revalidate",       P_BOOL,    P_LOCAL,  &sDefault.bRevalidate,      NULL,   NULL},
  577.   {"default case",     P_ENUM, P_LOCAL,  &sDefault.iDefaultCase,        NULL,   enum_case},
  578.   {"case sensitive",   P_BOOL,    P_LOCAL,  &sDefault.bCaseSensitive,   NULL,   NULL},
  579.   {"casesignames",     P_BOOL,    P_LOCAL,  &sDefault.bCaseSensitive,   NULL,   NULL},
  580.   {"preserve case",    P_BOOL,    P_LOCAL,  &sDefault.bCasePreserve,    NULL,   NULL},
  581.   {"short preserve case",P_BOOL,  P_LOCAL,  &sDefault.bShortCasePreserve,NULL,   NULL},
  582.   {"mangle case",      P_BOOL,    P_LOCAL,  &sDefault.bCaseMangle,      NULL,   NULL},
  583.   {"mangling char",    P_CHAR,    P_LOCAL,  &sDefault.magic_char,       NULL,   NULL},
  584.   {"browseable",       P_BOOL,    P_LOCAL,  &sDefault.bBrowseable,      NULL,   NULL},
  585.   {"browsable",        P_BOOL,    P_LOCAL,  &sDefault.bBrowseable,      NULL,   NULL},
  586.   {"available",        P_BOOL,    P_LOCAL,  &sDefault.bAvailable,       NULL,   NULL},
  587.   {"path",             P_STRING,  P_LOCAL,  &sDefault.szPath,           NULL,   NULL},
  588.   {"directory",        P_STRING,  P_LOCAL,  &sDefault.szPath,           NULL,   NULL},
  589.   {"username",         P_STRING,  P_LOCAL,  &sDefault.szUsername,       NULL,   NULL},
  590.   {"user",             P_STRING,  P_LOCAL,  &sDefault.szUsername,       NULL,   NULL},
  591.   {"users",            P_STRING,  P_LOCAL,  &sDefault.szUsername,       NULL,   NULL},
  592.   {"guest account",    P_STRING,  P_LOCAL,  &sDefault.szGuestaccount,   NULL,   NULL},
  593.   {"invalid users",    P_STRING,  P_LOCAL,  &sDefault.szInvalidUsers,   NULL,   NULL},
  594.   {"valid users",      P_STRING,  P_LOCAL,  &sDefault.szValidUsers,     NULL,   NULL},
  595.   {"admin users",      P_STRING,  P_LOCAL,  &sDefault.szAdminUsers,     NULL,   NULL},
  596.   {"read list",        P_STRING,  P_LOCAL,  &sDefault.readlist,         NULL,   NULL},
  597.   {"write list",       P_STRING,  P_LOCAL,  &sDefault.writelist,        NULL,   NULL},
  598.   {"volume",           P_STRING,  P_LOCAL,  &sDefault.volume,           NULL,   NULL},
  599.   {"force user",       P_STRING,  P_LOCAL,  &sDefault.force_user,       NULL,   NULL},
  600.   {"force group",      P_STRING,  P_LOCAL,  &sDefault.force_group,      NULL,   NULL},
  601.   {"group",            P_STRING,  P_LOCAL,  &sDefault.force_group,      NULL,   NULL},
  602.   {"read only",        P_BOOL,    P_LOCAL,  &sDefault.bRead_only,       NULL,   NULL},
  603.   {"write ok",         P_BOOLREV, P_LOCAL,  &sDefault.bRead_only,       NULL,   NULL},
  604.   {"writeable",        P_BOOLREV, P_LOCAL,  &sDefault.bRead_only,       NULL,   NULL},
  605.   {"writable",         P_BOOLREV, P_LOCAL,  &sDefault.bRead_only,       NULL,   NULL},
  606.   {"max connections",  P_INTEGER, P_LOCAL,  &sDefault.iMaxConnections,  NULL,   NULL},
  607.   {"min print space",  P_INTEGER, P_LOCAL,  &sDefault.iMinPrintSpace,   NULL,   NULL},
  608.   {"create mask",      P_OCTAL,   P_LOCAL,  &sDefault.iCreate_mask,     NULL,   NULL},
  609.   {"create mode",      P_OCTAL,   P_LOCAL,  &sDefault.iCreate_mask,     NULL,   NULL},
  610.   {"force create mode",P_OCTAL,   P_LOCAL,  &sDefault.iCreate_force_mode,     NULL,   NULL},
  611.   {"directory mask",   P_OCTAL,   P_LOCAL,  &sDefault.iDir_mask,        NULL,   NULL},
  612.   {"directory mode",   P_OCTAL,   P_LOCAL,  &sDefault.iDir_mask,        NULL,   NULL},
  613.   {"force directory mode",   P_OCTAL,   P_LOCAL,  &sDefault.iDir_force_mode,        NULL,   NULL},
  614.   {"set directory",    P_BOOLREV, P_LOCAL,  &sDefault.bNo_set_dir,      NULL,   NULL},
  615.   {"status",           P_BOOL,    P_LOCAL,  &sDefault.status,           NULL,   NULL},
  616.   {"hide dot files",   P_BOOL,    P_LOCAL,  &sDefault.bHideDotFiles,    NULL,   NULL},
  617.   {"delete veto files",P_BOOL,    P_LOCAL,  &sDefault.bDeleteVetoFiles, NULL,   NULL},
  618.   {"veto files",       P_STRING,  P_LOCAL,  &sDefault.szVetoFiles,      NULL,   NULL},
  619.   {"hide files",       P_STRING,  P_LOCAL,  &sDefault.szHideFiles,      NULL,   NULL},
  620.   {"veto oplock files",P_STRING,  P_LOCAL,  &sDefault.szVetoOplockFiles,NULL,   NULL},
  621.   {"guest only",       P_BOOL,    P_LOCAL,  &sDefault.bGuest_only,      NULL,   NULL},
  622.   {"only guest",       P_BOOL,    P_LOCAL,  &sDefault.bGuest_only,      NULL,   NULL},
  623.   {"guest ok",         P_BOOL,    P_LOCAL,  &sDefault.bGuest_ok,        NULL,   NULL},
  624.   {"public",           P_BOOL,    P_LOCAL,  &sDefault.bGuest_ok,        NULL,   NULL},
  625.   {"print ok",         P_BOOL,    P_LOCAL,  &sDefault.bPrint_ok,        NULL,   NULL},
  626.   {"printable",        P_BOOL,    P_LOCAL,  &sDefault.bPrint_ok,        NULL,   NULL},
  627.   {"postscript",       P_BOOL,    P_LOCAL,  &sDefault.bPostscript,      NULL,   NULL},
  628.   {"map system",       P_BOOL,    P_LOCAL,  &sDefault.bMap_system,      NULL,   NULL},
  629.   {"map hidden",       P_BOOL,    P_LOCAL,  &sDefault.bMap_hidden,      NULL,   NULL},
  630.   {"map archive",      P_BOOL,    P_LOCAL,  &sDefault.bMap_archive,     NULL,   NULL},
  631.   {"locking",          P_BOOL,    P_LOCAL,  &sDefault.bLocking,         NULL,   NULL},
  632.   {"strict locking",   P_BOOL,    P_LOCAL,  &sDefault.bStrictLocking,   NULL,   NULL},
  633.   {"share modes",      P_BOOL,    P_LOCAL,  &sDefault.bShareModes,      NULL,   NULL},
  634.   {"oplocks",          P_BOOL,    P_LOCAL,  &sDefault.bOpLocks,         NULL,   NULL},
  635.   {"only user",        P_BOOL,    P_LOCAL,  &sDefault.bOnlyUser,        NULL,   NULL},
  636.   {"wide links",       P_BOOL,    P_LOCAL,  &sDefault.bWidelinks,       NULL,   NULL},
  637.   {"follow symlinks",  P_BOOL,    P_LOCAL,  &sDefault.bSymlinks,        NULL,   NULL},
  638.   {"sync always",      P_BOOL,    P_LOCAL,  &sDefault.bSyncAlways,      NULL,   NULL},
  639.   {"strict sync",      P_BOOL,    P_LOCAL,  &sDefault.bStrictSync,      NULL,   NULL},
  640.   {"mangled names",    P_BOOL,    P_LOCAL,  &sDefault.bMangledNames,    NULL,   NULL},
  641.   {"fake oplocks",     P_BOOL,    P_LOCAL,  &sDefault.bFakeOplocks,     NULL,   NULL},
  642.   {"printing",         P_ENUM,    P_LOCAL,  &sDefault.iPrinting,        NULL,   enum_printing},
  643.   {"print command",    P_STRING,  P_LOCAL,  &sDefault.szPrintcommand,   NULL,   NULL},
  644.   {"lpq command",      P_STRING,  P_LOCAL,  &sDefault.szLpqcommand,     NULL,   NULL},
  645.   {"lprm command",     P_STRING,  P_LOCAL,  &sDefault.szLprmcommand,    NULL,   NULL},
  646.   {"lppause command",  P_STRING,  P_LOCAL,  &sDefault.szLppausecommand, NULL,   NULL},
  647.   {"lpresume command", P_STRING,  P_LOCAL,  &sDefault.szLpresumecommand,NULL,   NULL},
  648.   {"queuepause command", P_STRING, P_LOCAL, &sDefault.szQueuepausecommand, NULL, NULL},
  649.   {"queueresume command", P_STRING, P_LOCAL, &sDefault.szQueueresumecommand, NULL, NULL},
  650.   {"printer",          P_STRING,  P_LOCAL,  &sDefault.szPrintername,    NULL,   NULL},
  651.   {"printer name",     P_STRING,  P_LOCAL,  &sDefault.szPrintername,    NULL,   NULL},
  652.   {"printer driver",   P_STRING,  P_LOCAL,  &sDefault.szPrinterDriver,  NULL,   NULL},
  653.   {"printer driver location",   P_STRING,  P_LOCAL,  &sDefault.szPrinterDriverLocation,  NULL,   NULL},
  654.   {"hosts allow",      P_STRING,  P_LOCAL,  &sDefault.szHostsallow,     NULL,   NULL},
  655.   {"allow hosts",      P_STRING,  P_LOCAL,  &sDefault.szHostsallow,     NULL,   NULL},
  656.   {"hosts deny",       P_STRING,  P_LOCAL,  &sDefault.szHostsdeny,      NULL,   NULL},
  657.   {"deny hosts",       P_STRING,  P_LOCAL,  &sDefault.szHostsdeny,      NULL,   NULL},
  658.   {"dont descend",     P_STRING,  P_LOCAL,  &sDefault.szDontdescend,    NULL,   NULL},
  659.   {"magic script",     P_STRING,  P_LOCAL,  &sDefault.szMagicScript,    NULL,   NULL},
  660.   {"magic output",     P_STRING,  P_LOCAL,  &sDefault.szMagicOutput,    NULL,   NULL},
  661.   {"mangled map",      P_STRING,  P_LOCAL,  &sDefault.szMangledMap,     NULL,   NULL},
  662.   {"delete readonly",  P_BOOL,    P_LOCAL,  &sDefault.bDeleteReadonly,  NULL,   NULL},
  663.   {"dos filetimes",    P_BOOL,    P_LOCAL,  &sDefault.bDosFiletimes,    NULL,   NULL},
  664.   {"dos filetime resolution",   P_BOOL,    P_LOCAL,  &sDefault.bDosFiletimeResolution,    NULL,   NULL},
  665.   {"fake directory create times",    P_BOOL,    P_LOCAL,  &sDefault.bFakeDirCreateTimes, NULL,   NULL},
  666.  
  667.   {NULL,               P_BOOL,    P_NONE,   NULL,                       NULL,   NULL}
  668. };
  669.  
  670.  
  671.  
  672. /***************************************************************************
  673. Initialise the global parameter structure.
  674. ***************************************************************************/
  675. static void init_globals(void)
  676. {
  677.   static BOOL done_init = False;
  678.   pstring s;
  679.  
  680.   if (!done_init)
  681.     {
  682.       int i;
  683.       bzero((void *)&Globals,sizeof(Globals));
  684.  
  685.       for (i = 0; parm_table[i].label; i++) 
  686.     if ((parm_table[i].type == P_STRING ||
  687.          parm_table[i].type == P_USTRING) && 
  688.         parm_table[i].ptr)
  689.       string_init(parm_table[i].ptr,"");
  690.  
  691.       string_set(&sDefault.szGuestaccount, GUEST_ACCOUNT);
  692.       string_set(&sDefault.szPrinterDriver, "NULL");
  693.  
  694.       done_init = True;
  695.     }
  696.  
  697.  
  698.   DEBUG(3,("Initialising global parameters\n"));
  699.  
  700. #ifdef SMB_PASSWD_FILE
  701.   string_set(&Globals.szSMBPasswdFile, SMB_PASSWD_FILE);
  702. #endif
  703.   string_set(&Globals.szPasswdChat,"*old*password* %o\\n *new*password* %n\\n *new*password* %n\\n *changed*");
  704.   string_set(&Globals.szWorkGroup, WORKGROUP);
  705.   string_set(&Globals.szPasswdProgram, SMB_PASSWD);
  706.   string_set(&Globals.szPrintcapname, PRINTCAP_NAME);
  707.   string_set(&Globals.szDriverFile, DRIVERFILE);
  708.   string_set(&Globals.szLockDir, LOCKDIR);
  709.   string_set(&Globals.szRootdir, "/");
  710.   string_set(&Globals.szSmbrun, SMBRUN);
  711.   string_set(&Globals.szSocketAddress, "0.0.0.0");
  712.   slprintf(s,sizeof(s)-1,"Samba %s",VERSION);
  713.   string_set(&Globals.szServerString,s);
  714.   slprintf(s,sizeof(s)-1,"%d.%d", DEFAULT_MAJOR_VERSION, DEFAULT_MINOR_VERSION);
  715.   string_set(&Globals.szAnnounceVersion,s);
  716.  
  717.   string_set(&Globals.szLogonDrive, "");
  718.   /* %N is the NIS auto.home server if -DAUTOHOME is used, else same as %L */
  719.   string_set(&Globals.szLogonHome, "\\\\%N\\%U");
  720.   string_set(&Globals.szLogonPath, "\\\\%N\\%U\\profile");
  721.  
  722.   string_set(&Globals.szNameResolveOrder, "lmhosts host wins bcast");
  723.  
  724.   Globals.bLoadPrinters = True;
  725.   Globals.bUseRhosts = False;
  726.   Globals.max_packet = 65535;
  727.   Globals.mangled_stack = 50;
  728.   Globals.max_xmit = 65535;
  729.   Globals.max_mux = 50; /* This is *needed* for profile support. */
  730.   Globals.lpqcachetime = 10;
  731.   Globals.pwordlevel = 0;
  732.   Globals.unamelevel = 0;
  733.   Globals.deadtime = 0;
  734.   Globals.max_log_size = 5000;
  735.   Globals.maxprotocol = PROTOCOL_NT1;
  736.   Globals.security = SEC_SHARE;
  737.   Globals.bEncryptPasswords = False;
  738.   Globals.bUpdateEncrypt = False;
  739.   Globals.bReadRaw = True;
  740.   Globals.bWriteRaw = True;
  741.   Globals.bReadPrediction = False;
  742.   Globals.bReadbmpx = True;
  743.   Globals.bNullPasswords = False;
  744.   Globals.bStripDot = False;
  745.   Globals.syslog = 1;
  746.   Globals.bSyslogOnly = False;
  747.   Globals.os_level = 0;
  748.   Globals.max_ttl = 60*60*4; /* 4 hours default */
  749.   Globals.max_wins_ttl = 60*60*24*3; /* 3 days default */
  750.   Globals.min_wins_ttl = 60*60*6; /* 6 hours default */
  751.   Globals.ReadSize = 16*1024;
  752.   Globals.lm_announce = 2;   /* = Auto: send only if LM clients found */
  753.   Globals.lm_interval = 60;
  754.   Globals.shmem_size = SHMEM_SIZE;
  755.   Globals.announce_as = ANNOUNCE_AS_NT;
  756.   Globals.bUnixRealname = False;
  757. #if (defined(NETGROUP) && defined(AUTOMOUNT))
  758.   Globals.bNISHomeMap = False;
  759. #ifdef NISPLUS
  760.   string_set(&Globals.szNISHomeMapName, "auto_home.org_dir");
  761. #else
  762.   string_set(&Globals.szNISHomeMapName, "auto.home");
  763. #endif
  764. #endif
  765.   Globals.client_code_page = DEFAULT_CLIENT_CODE_PAGE;
  766.   Globals.bTimeServer = False;
  767.   Globals.bBindInterfacesOnly = False;
  768.   Globals.bNetWkstaUserLogon = False; /* This is now set to false by default as
  769.                                          the code in password.c protects us from this bug. */
  770.   Globals.bUnixPasswdSync = False;
  771.   Globals.bPasswdChatDebug = False;
  772.   Globals.bOleLockingCompat = True;
  773.  
  774. /* these parameters are set to defaults that are more appropriate
  775.    for the increasing samba install base:
  776.  
  777.    as a member of the workgroup, that will possibly become a
  778.    _local_ master browser (lm = True).  this is opposed to a forced
  779.    local master browser startup (pm = True).
  780.  
  781.    doesn't provide WINS server service by default (wsupp = False),
  782.    and doesn't provide domain master browser services by default, either.
  783.  
  784. */
  785.  
  786.   Globals.bPreferredMaster = False;
  787.   Globals.bLocalMaster = True;
  788.   Globals.bDomainMaster = False;
  789.   Globals.bDomainLogons = False;
  790.   Globals.bBrowseList = True;
  791.   Globals.bWINSsupport = False;
  792.   Globals.bWINSproxy = False;
  793.  
  794.   Globals.bDNSproxy = True;
  795.  
  796.   /*
  797.    * This must be done last as it checks the value in 
  798.    * client_code_page.
  799.    */
  800.  
  801.   interpret_coding_system(KANJI);
  802. }
  803.  
  804. /***************************************************************************
  805. check if a string is initialised and if not then initialise it
  806. ***************************************************************************/
  807. static void string_initial(char **s,char *v)
  808. {
  809.   if (!*s || !**s)
  810.     string_init(s,v);
  811. }
  812.  
  813.  
  814. /***************************************************************************
  815. Initialise the sDefault parameter structure.
  816. ***************************************************************************/
  817. static void init_locals(void)
  818. {
  819.   /* choose defaults depending on the type of printing */
  820.   switch (sDefault.iPrinting)
  821.     {
  822.     case PRINT_BSD:
  823.     case PRINT_AIX:
  824.     case PRINT_LPRNG:
  825.     case PRINT_PLP:
  826.       string_initial(&sDefault.szLpqcommand,"lpq -P%p");
  827.       string_initial(&sDefault.szLprmcommand,"lprm -P%p %j");
  828.       string_initial(&sDefault.szPrintcommand,"lpr -r -P%p %s");
  829.       break;
  830.  
  831.     case PRINT_SYSV:
  832.     case PRINT_HPUX:
  833.       string_initial(&sDefault.szLpqcommand,"lpstat -o%p");
  834.       string_initial(&sDefault.szLprmcommand,"cancel %p-%j");
  835.       string_initial(&sDefault.szPrintcommand,"lp -c -d%p %s; rm %s");
  836. #ifdef SVR4
  837.       string_initial(&sDefault.szLppausecommand,"lp -i %p-%j -H hold");
  838.       string_initial(&sDefault.szLpresumecommand,"lp -i %p-%j -H resume");
  839.       string_initial(&sDefault.szQueuepausecommand, "lpc stop %p");
  840.       string_initial(&sDefault.szQueueresumecommand, "lpc start %p");
  841. #else /* SVR4 */
  842.       string_initial(&sDefault.szQueuepausecommand, "disable %p");
  843.       string_initial(&sDefault.szQueueresumecommand, "enable %p");
  844. #endif /* SVR4 */
  845.       break;
  846.  
  847.     case PRINT_QNX:
  848.       string_initial(&sDefault.szLpqcommand,"lpq -P%p");
  849.       string_initial(&sDefault.szLprmcommand,"lprm -P%p %j");
  850.       string_initial(&sDefault.szPrintcommand,"lp -r -P%p %s");
  851.       break;
  852.  
  853.     case PRINT_SOFTQ:
  854.       string_initial(&sDefault.szLpqcommand,"qstat -l -d%p");
  855.       string_initial(&sDefault.szLprmcommand,"qstat -s -j%j -c");
  856.       string_initial(&sDefault.szPrintcommand,"lp -d%p -s %s; rm %s");
  857.       string_initial(&sDefault.szLppausecommand,"qstat -s -j%j -h");
  858.       string_initial(&sDefault.szLpresumecommand,"qstat -s -j%j -r");
  859.       break;
  860.       
  861.     }
  862. }
  863.  
  864.  
  865. /******************************************************************* a
  866. convenience routine to grab string parameters into a rotating buffer,
  867. and run standard_sub_basic on them. The buffers can be written to by
  868. callers without affecting the source string.
  869. ********************************************************************/
  870. char *lp_string(char *s)
  871. {
  872.   static char *bufs[10];
  873.   static int buflen[10];
  874.   static int next = -1;  
  875.   char *ret;
  876.   int i;
  877.   int len = s?strlen(s):0;
  878.  
  879.   if (next == -1) {
  880.     /* initialisation */
  881.     for (i=0;i<10;i++) {
  882.       bufs[i] = NULL;
  883.       buflen[i] = 0;
  884.     }
  885.     next = 0;
  886.   }
  887.  
  888.   len = MAX(len+100,sizeof(pstring)); /* the +100 is for some
  889.                      substitution room */
  890.  
  891.   if (buflen[next] != len) {
  892.     buflen[next] = len;
  893.     if (bufs[next]) free(bufs[next]);
  894.     bufs[next] = (char *)malloc(len);
  895.     if (!bufs[next]) {
  896.       DEBUG(0,("out of memory in lp_string()"));
  897.       exit(1);
  898.     }
  899.   } 
  900.  
  901.   ret = &bufs[next][0];
  902.   next = (next+1)%10;
  903.  
  904.   if (!s) 
  905.     *ret = 0;
  906.   else
  907.     StrCpy(ret,s);
  908.  
  909.   trim_string(ret, "\"", "\"");
  910.  
  911.   standard_sub_basic(ret);
  912.   return(ret);
  913. }
  914.  
  915.  
  916. /*
  917.    In this section all the functions that are used to access the 
  918.    parameters from the rest of the program are defined 
  919. */
  920.  
  921. #define FN_GLOBAL_STRING(fn_name,ptr) \
  922.  char *fn_name(void) {return(lp_string(*(char **)(ptr) ? *(char **)(ptr) : ""));}
  923. #define FN_GLOBAL_BOOL(fn_name,ptr) \
  924.  BOOL fn_name(void) {return(*(BOOL *)(ptr));}
  925. #define FN_GLOBAL_CHAR(fn_name,ptr) \
  926.  char fn_name(void) {return(*(char *)(ptr));}
  927. #define FN_GLOBAL_INTEGER(fn_name,ptr) \
  928.  int fn_name(void) {return(*(int *)(ptr));}
  929.  
  930. #define FN_LOCAL_STRING(fn_name,val) \
  931.  char *fn_name(int i) {return(lp_string((LP_SNUM_OK(i)&&pSERVICE(i)->val)?pSERVICE(i)->val : sDefault.val));}
  932. #define FN_LOCAL_BOOL(fn_name,val) \
  933.  BOOL fn_name(int i) {return(LP_SNUM_OK(i)? pSERVICE(i)->val : sDefault.val);}
  934. #define FN_LOCAL_CHAR(fn_name,val) \
  935.  char fn_name(int i) {return(LP_SNUM_OK(i)? pSERVICE(i)->val : sDefault.val);}
  936. #define FN_LOCAL_INTEGER(fn_name,val) \
  937.  int fn_name(int i) {return(LP_SNUM_OK(i)? pSERVICE(i)->val : sDefault.val);}
  938.  
  939. FN_GLOBAL_STRING(lp_logfile,&Globals.szLogFile)
  940. FN_GLOBAL_STRING(lp_smbrun,&Globals.szSmbrun)
  941. FN_GLOBAL_STRING(lp_configfile,&Globals.szConfigFile)
  942. FN_GLOBAL_STRING(lp_smb_passwd_file,&Globals.szSMBPasswdFile)
  943. FN_GLOBAL_STRING(lp_serverstring,&Globals.szServerString)
  944. FN_GLOBAL_STRING(lp_printcapname,&Globals.szPrintcapname)
  945. FN_GLOBAL_STRING(lp_lockdir,&Globals.szLockDir)
  946. FN_GLOBAL_STRING(lp_rootdir,&Globals.szRootdir)
  947. FN_GLOBAL_STRING(lp_defaultservice,&Globals.szDefaultService)
  948. FN_GLOBAL_STRING(lp_msg_command,&Globals.szMsgCommand)
  949. FN_GLOBAL_STRING(lp_dfree_command,&Globals.szDfree)
  950. FN_GLOBAL_STRING(lp_hosts_equiv,&Globals.szHostsEquiv)
  951. FN_GLOBAL_STRING(lp_auto_services,&Globals.szAutoServices)
  952. FN_GLOBAL_STRING(lp_passwd_program,&Globals.szPasswdProgram)
  953. FN_GLOBAL_STRING(lp_passwd_chat,&Globals.szPasswdChat)
  954. FN_GLOBAL_STRING(lp_passwordserver,&Globals.szPasswordServer)
  955. FN_GLOBAL_STRING(lp_name_resolve_order,&Globals.szNameResolveOrder)
  956. FN_GLOBAL_STRING(lp_workgroup,&Globals.szWorkGroup)
  957. FN_GLOBAL_STRING(lp_username_map,&Globals.szUsernameMap)
  958. FN_GLOBAL_STRING(lp_character_set,&Globals.szCharacterSet) 
  959. FN_GLOBAL_STRING(lp_logon_script,&Globals.szLogonScript) 
  960. FN_GLOBAL_STRING(lp_logon_path,&Globals.szLogonPath) 
  961. FN_GLOBAL_STRING(lp_logon_drive,&Globals.szLogonDrive) 
  962. FN_GLOBAL_STRING(lp_logon_home,&Globals.szLogonHome) 
  963. FN_GLOBAL_STRING(lp_remote_announce,&Globals.szRemoteAnnounce) 
  964. FN_GLOBAL_STRING(lp_remote_browse_sync,&Globals.szRemoteBrowseSync) 
  965. FN_GLOBAL_STRING(lp_wins_server,&Globals.szWINSserver)
  966. FN_GLOBAL_STRING(lp_interfaces,&Globals.szInterfaces)
  967. FN_GLOBAL_STRING(lp_socket_address,&Globals.szSocketAddress)
  968. FN_GLOBAL_STRING(lp_nis_home_map_name,&Globals.szNISHomeMapName)
  969. FN_GLOBAL_STRING(lp_announce_version,&Globals.szAnnounceVersion)
  970. FN_GLOBAL_STRING(lp_netbios_aliases,&Globals.szNetbiosAliases)
  971. FN_GLOBAL_STRING(lp_driverfile,&Globals.szDriverFile)
  972.  
  973. #ifdef NTDOMAIN
  974. FN_GLOBAL_STRING(lp_domain_sid,&Globals.szDomainSID)
  975. FN_GLOBAL_STRING(lp_domain_other_sids,&Globals.szDomainOtherSIDs)
  976. FN_GLOBAL_STRING(lp_domain_groups,&Globals.szDomainGroups)
  977. FN_GLOBAL_STRING(lp_domain_admin_users,&Globals.szDomainAdminUsers)
  978. FN_GLOBAL_STRING(lp_domain_guest_users,&Globals.szDomainGuestUsers)
  979. FN_GLOBAL_STRING(lp_domain_hostsallow,&Globals.szDomainHostsallow)
  980. FN_GLOBAL_STRING(lp_domain_hostsdeny,&Globals.szDomainHostsdeny)
  981. #endif /* NTDOMAIN */
  982.  
  983. FN_GLOBAL_BOOL(lp_dns_proxy,&Globals.bDNSproxy)
  984. FN_GLOBAL_BOOL(lp_wins_support,&Globals.bWINSsupport)
  985. FN_GLOBAL_BOOL(lp_we_are_a_wins_server,&Globals.bWINSsupport)
  986. FN_GLOBAL_BOOL(lp_wins_proxy,&Globals.bWINSproxy)
  987. FN_GLOBAL_BOOL(lp_local_master,&Globals.bLocalMaster)
  988. FN_GLOBAL_BOOL(lp_domain_controller,&Globals.bDomainController)
  989. FN_GLOBAL_BOOL(lp_domain_master,&Globals.bDomainMaster)
  990. FN_GLOBAL_BOOL(lp_domain_logons,&Globals.bDomainLogons)
  991. FN_GLOBAL_BOOL(lp_preferred_master,&Globals.bPreferredMaster)
  992. FN_GLOBAL_BOOL(lp_load_printers,&Globals.bLoadPrinters)
  993. FN_GLOBAL_BOOL(lp_use_rhosts,&Globals.bUseRhosts)
  994. FN_GLOBAL_BOOL(lp_getwdcache,&use_getwd_cache)
  995. FN_GLOBAL_BOOL(lp_readprediction,&Globals.bReadPrediction)
  996. FN_GLOBAL_BOOL(lp_readbmpx,&Globals.bReadbmpx)
  997. FN_GLOBAL_BOOL(lp_readraw,&Globals.bReadRaw)
  998. FN_GLOBAL_BOOL(lp_writeraw,&Globals.bWriteRaw)
  999. FN_GLOBAL_BOOL(lp_null_passwords,&Globals.bNullPasswords)
  1000. FN_GLOBAL_BOOL(lp_strip_dot,&Globals.bStripDot)
  1001. FN_GLOBAL_BOOL(lp_encrypted_passwords,&Globals.bEncryptPasswords)
  1002. FN_GLOBAL_BOOL(lp_update_encrypted,&Globals.bUpdateEncrypt)
  1003. FN_GLOBAL_BOOL(lp_syslog_only,&Globals.bSyslogOnly)
  1004. FN_GLOBAL_BOOL(lp_browse_list,&Globals.bBrowseList)
  1005. FN_GLOBAL_BOOL(lp_unix_realname,&Globals.bUnixRealname)
  1006. FN_GLOBAL_BOOL(lp_nis_home_map,&Globals.bNISHomeMap)
  1007. FN_GLOBAL_BOOL(lp_time_server,&Globals.bTimeServer)
  1008. FN_GLOBAL_BOOL(lp_bind_interfaces_only,&Globals.bBindInterfacesOnly)
  1009. FN_GLOBAL_BOOL(lp_net_wksta_user_logon,&Globals.bNetWkstaUserLogon)
  1010. FN_GLOBAL_BOOL(lp_unix_password_sync,&Globals.bUnixPasswdSync)
  1011. FN_GLOBAL_BOOL(lp_passwd_chat_debug,&Globals.bPasswdChatDebug)
  1012. FN_GLOBAL_BOOL(lp_ole_locking_compat,&Globals.bOleLockingCompat)
  1013.  
  1014. FN_GLOBAL_INTEGER(lp_os_level,&Globals.os_level)
  1015. FN_GLOBAL_INTEGER(lp_max_ttl,&Globals.max_ttl)
  1016. FN_GLOBAL_INTEGER(lp_max_wins_ttl,&Globals.max_wins_ttl)
  1017. FN_GLOBAL_INTEGER(lp_min_wins_ttl,&Globals.max_wins_ttl)
  1018. FN_GLOBAL_INTEGER(lp_max_log_size,&Globals.max_log_size)
  1019. FN_GLOBAL_INTEGER(lp_mangledstack,&Globals.mangled_stack)
  1020. FN_GLOBAL_INTEGER(lp_maxxmit,&Globals.max_xmit)
  1021. FN_GLOBAL_INTEGER(lp_maxmux,&Globals.max_mux)
  1022. FN_GLOBAL_INTEGER(lp_maxpacket,&Globals.max_packet)
  1023. FN_GLOBAL_INTEGER(lp_keepalive,&keepalive)
  1024. FN_GLOBAL_INTEGER(lp_passwordlevel,&Globals.pwordlevel)
  1025. FN_GLOBAL_INTEGER(lp_usernamelevel,&Globals.unamelevel)
  1026. FN_GLOBAL_INTEGER(lp_readsize,&Globals.ReadSize)
  1027. FN_GLOBAL_INTEGER(lp_shmem_size,&Globals.shmem_size)
  1028. FN_GLOBAL_INTEGER(lp_deadtime,&Globals.deadtime)
  1029. FN_GLOBAL_INTEGER(lp_maxprotocol,&Globals.maxprotocol)
  1030. FN_GLOBAL_INTEGER(lp_security,&Globals.security)
  1031. FN_GLOBAL_INTEGER(lp_maxdisksize,&Globals.maxdisksize)
  1032. FN_GLOBAL_INTEGER(lp_lpqcachetime,&Globals.lpqcachetime)
  1033. FN_GLOBAL_INTEGER(lp_syslog,&Globals.syslog)
  1034. FN_GLOBAL_INTEGER(lp_client_code_page,&Globals.client_code_page)
  1035. FN_GLOBAL_INTEGER(lp_announce_as,&Globals.announce_as)
  1036. FN_GLOBAL_INTEGER(lp_lm_announce,&Globals.lm_announce)
  1037. FN_GLOBAL_INTEGER(lp_lm_interval,&Globals.lm_interval)
  1038.  
  1039. FN_LOCAL_STRING(lp_preexec,szPreExec)
  1040. FN_LOCAL_STRING(lp_postexec,szPostExec)
  1041. FN_LOCAL_STRING(lp_rootpreexec,szRootPreExec)
  1042. FN_LOCAL_STRING(lp_rootpostexec,szRootPostExec)
  1043. FN_LOCAL_STRING(lp_servicename,szService)
  1044. FN_LOCAL_STRING(lp_pathname,szPath)
  1045. FN_LOCAL_STRING(lp_dontdescend,szDontdescend)
  1046. FN_LOCAL_STRING(lp_username,szUsername)
  1047. FN_LOCAL_STRING(lp_guestaccount,szGuestaccount)
  1048. FN_LOCAL_STRING(lp_invalid_users,szInvalidUsers)
  1049. FN_LOCAL_STRING(lp_valid_users,szValidUsers)
  1050. FN_LOCAL_STRING(lp_admin_users,szAdminUsers)
  1051. FN_LOCAL_STRING(lp_printcommand,szPrintcommand)
  1052. FN_LOCAL_STRING(lp_lpqcommand,szLpqcommand)
  1053. FN_LOCAL_STRING(lp_lprmcommand,szLprmcommand)
  1054. FN_LOCAL_STRING(lp_lppausecommand,szLppausecommand)
  1055. FN_LOCAL_STRING(lp_lpresumecommand,szLpresumecommand)
  1056. FN_LOCAL_STRING(lp_queuepausecommand,szQueuepausecommand)
  1057. FN_LOCAL_STRING(lp_queueresumecommand,szQueueresumecommand)
  1058. FN_LOCAL_STRING(lp_printername,szPrintername)
  1059. FN_LOCAL_STRING(lp_printerdriver,szPrinterDriver)
  1060. FN_LOCAL_STRING(lp_hostsallow,szHostsallow)
  1061. FN_LOCAL_STRING(lp_hostsdeny,szHostsdeny)
  1062. FN_LOCAL_STRING(lp_magicscript,szMagicScript)
  1063. FN_LOCAL_STRING(lp_magicoutput,szMagicOutput)
  1064. FN_LOCAL_STRING(lp_comment,comment)
  1065. FN_LOCAL_STRING(lp_force_user,force_user)
  1066. FN_LOCAL_STRING(lp_force_group,force_group)
  1067. FN_LOCAL_STRING(lp_readlist,readlist)
  1068. FN_LOCAL_STRING(lp_writelist,writelist)
  1069. FN_LOCAL_STRING(lp_volume,volume)
  1070. FN_LOCAL_STRING(lp_mangled_map,szMangledMap)
  1071. FN_LOCAL_STRING(lp_veto_files,szVetoFiles)
  1072. FN_LOCAL_STRING(lp_hide_files,szHideFiles)
  1073. FN_LOCAL_STRING(lp_veto_oplocks,szVetoOplockFiles)
  1074. FN_LOCAL_STRING(lp_driverlocation,szPrinterDriverLocation)
  1075.  
  1076. FN_LOCAL_BOOL(lp_alternate_permissions,bAlternatePerm)
  1077. FN_LOCAL_BOOL(lp_revalidate,bRevalidate)
  1078. FN_LOCAL_BOOL(lp_casesensitive,bCaseSensitive)
  1079. FN_LOCAL_BOOL(lp_preservecase,bCasePreserve)
  1080. FN_LOCAL_BOOL(lp_shortpreservecase,bShortCasePreserve)
  1081. FN_LOCAL_BOOL(lp_casemangle,bCaseMangle)
  1082. FN_LOCAL_BOOL(lp_status,status)
  1083. FN_LOCAL_BOOL(lp_hide_dot_files,bHideDotFiles)
  1084. FN_LOCAL_BOOL(lp_browseable,bBrowseable)
  1085. FN_LOCAL_BOOL(lp_readonly,bRead_only)
  1086. FN_LOCAL_BOOL(lp_no_set_dir,bNo_set_dir)
  1087. FN_LOCAL_BOOL(lp_guest_ok,bGuest_ok)
  1088. FN_LOCAL_BOOL(lp_guest_only,bGuest_only)
  1089. FN_LOCAL_BOOL(lp_print_ok,bPrint_ok)
  1090. FN_LOCAL_BOOL(lp_postscript,bPostscript)
  1091. FN_LOCAL_BOOL(lp_map_hidden,bMap_hidden)
  1092. FN_LOCAL_BOOL(lp_map_archive,bMap_archive)
  1093. FN_LOCAL_BOOL(lp_locking,bLocking)
  1094. FN_LOCAL_BOOL(lp_strict_locking,bStrictLocking)
  1095. FN_LOCAL_BOOL(lp_share_modes,bShareModes)
  1096. FN_LOCAL_BOOL(lp_oplocks,bOpLocks)
  1097. FN_LOCAL_BOOL(lp_onlyuser,bOnlyUser)
  1098. FN_LOCAL_BOOL(lp_manglednames,bMangledNames)
  1099. FN_LOCAL_BOOL(lp_widelinks,bWidelinks)
  1100. FN_LOCAL_BOOL(lp_symlinks,bSymlinks)
  1101. FN_LOCAL_BOOL(lp_syncalways,bSyncAlways)
  1102. FN_LOCAL_BOOL(lp_strict_sync,bStrictSync)
  1103. FN_LOCAL_BOOL(lp_map_system,bMap_system)
  1104. FN_LOCAL_BOOL(lp_delete_readonly,bDeleteReadonly)
  1105. FN_LOCAL_BOOL(lp_fake_oplocks,bFakeOplocks)
  1106. FN_LOCAL_BOOL(lp_recursive_veto_delete,bDeleteVetoFiles)
  1107. FN_LOCAL_BOOL(lp_dos_filetimes,bDosFiletimes)
  1108. FN_LOCAL_BOOL(lp_dos_filetime_resolution,bDosFiletimeResolution)
  1109. FN_LOCAL_BOOL(lp_fake_dir_create_times,bFakeDirCreateTimes)
  1110.  
  1111. FN_LOCAL_INTEGER(lp_create_mode,iCreate_mask)
  1112. FN_LOCAL_INTEGER(lp_force_create_mode,iCreate_force_mode)
  1113. FN_LOCAL_INTEGER(lp_dir_mode,iDir_mask)
  1114. FN_LOCAL_INTEGER(lp_force_dir_mode,iDir_force_mode)
  1115. FN_LOCAL_INTEGER(lp_max_connections,iMaxConnections)
  1116. FN_LOCAL_INTEGER(lp_defaultcase,iDefaultCase)
  1117. FN_LOCAL_INTEGER(lp_minprintspace,iMinPrintSpace)
  1118. FN_LOCAL_INTEGER(lp_printing,iPrinting)
  1119.  
  1120. FN_LOCAL_CHAR(lp_magicchar,magic_char)
  1121.  
  1122.  
  1123.  
  1124. /* local prototypes */
  1125. static int    strwicmp( char *psz1, char *psz2 );
  1126. static int    map_parameter( char *pszParmName);
  1127. static BOOL   set_boolean( BOOL *pb, char *pszParmValue );
  1128. static int    getservicebyname(char *pszServiceName, service *pserviceDest);
  1129. static void   copy_service( service *pserviceDest, 
  1130.                             service *pserviceSource,
  1131.                             BOOL *pcopymapDest );
  1132. static BOOL   service_ok(int iService);
  1133. static BOOL   do_parameter(char *pszParmName, char *pszParmValue);
  1134. static BOOL   do_section(char *pszSectionName);
  1135. static void init_copymap(service *pservice);
  1136.  
  1137.  
  1138. /***************************************************************************
  1139. initialise a service to the defaults
  1140. ***************************************************************************/
  1141. static void init_service(service *pservice)
  1142. {
  1143.   bzero((char *)pservice,sizeof(service));
  1144.   copy_service(pservice,&sDefault,NULL);
  1145. }
  1146.  
  1147.  
  1148. /***************************************************************************
  1149. free the dynamically allocated parts of a service struct
  1150. ***************************************************************************/
  1151. static void free_service(service *pservice)
  1152. {
  1153.   int i;
  1154.   if (!pservice)
  1155.      return;
  1156.  
  1157.   if(pservice->szService)
  1158.     DEBUG(5,("free_service: Freeing service %s\n", pservice->szService));
  1159.  
  1160.   string_free(&pservice->szService);
  1161.   if (pservice->copymap)
  1162.   {
  1163.     free(pservice->copymap);
  1164.     pservice->copymap = NULL;
  1165.   }
  1166.  
  1167.   for (i=0;parm_table[i].label;i++)
  1168.     if ((parm_table[i].type == P_STRING ||
  1169.      parm_table[i].type == P_USTRING) &&
  1170.     parm_table[i].class == P_LOCAL)
  1171.       string_free((char **)(((char *)pservice) + PTR_DIFF(parm_table[i].ptr,&sDefault)));
  1172. }
  1173.  
  1174. /***************************************************************************
  1175. add a new service to the services array initialising it with the given 
  1176. service
  1177. ***************************************************************************/
  1178. static int add_a_service(service *pservice, char *name)
  1179. {
  1180.   int i;
  1181.   service tservice;
  1182.   int num_to_alloc = iNumServices+1;
  1183.  
  1184.   tservice = *pservice;
  1185.  
  1186.   /* it might already exist */
  1187.   if (name) 
  1188.     {
  1189.       i = getservicebyname(name,NULL);
  1190.       if (i >= 0)
  1191.     return(i);
  1192.     }
  1193.  
  1194.   /* find an invalid one */
  1195.   for (i=0;i<iNumServices;i++)
  1196.     if (!pSERVICE(i)->valid)
  1197.       break;
  1198.  
  1199.   /* if not, then create one */
  1200.   if (i == iNumServices)
  1201.     {
  1202.       ServicePtrs = (service **)Realloc(ServicePtrs,sizeof(service *)*num_to_alloc);
  1203.       if (ServicePtrs)
  1204.     pSERVICE(iNumServices) = (service *)malloc(sizeof(service));
  1205.  
  1206.       if (!ServicePtrs || !pSERVICE(iNumServices))
  1207.     return(-1);
  1208.  
  1209.       iNumServices++;
  1210.     }
  1211.   else
  1212.     free_service(pSERVICE(i));
  1213.  
  1214.   pSERVICE(i)->valid = True;
  1215.  
  1216.   init_service(pSERVICE(i));
  1217.   copy_service(pSERVICE(i),&tservice,NULL);
  1218.   if (name)
  1219.     string_set(&iSERVICE(i).szService,name);  
  1220.  
  1221.   return(i);
  1222. }
  1223.  
  1224. /***************************************************************************
  1225. add a new home service, with the specified home directory, defaults coming 
  1226. from service ifrom
  1227. ***************************************************************************/
  1228. BOOL lp_add_home(char *pszHomename, int iDefaultService, char *pszHomedir)
  1229. {
  1230.   int i = add_a_service(pSERVICE(iDefaultService),pszHomename);
  1231.  
  1232.   if (i < 0)
  1233.     return(False);
  1234.  
  1235.   if (!(*(iSERVICE(i).szPath)) || strequal(iSERVICE(i).szPath,lp_pathname(-1)))
  1236.     string_set(&iSERVICE(i).szPath,pszHomedir);
  1237.   if (!(*(iSERVICE(i).comment)))
  1238.     {
  1239.       pstring comment;
  1240.       slprintf(comment,sizeof(comment),
  1241.            "Home directory of %s",pszHomename);
  1242.       string_set(&iSERVICE(i).comment,comment);
  1243.     }
  1244.   iSERVICE(i).bAvailable = sDefault.bAvailable;
  1245.   iSERVICE(i).bBrowseable = sDefault.bBrowseable;
  1246.  
  1247.   DEBUG(3,("adding home directory %s at %s\n", pszHomename, pszHomedir));
  1248.  
  1249.   return(True);
  1250. }
  1251.  
  1252. /***************************************************************************
  1253. add a new service, based on an old one
  1254. ***************************************************************************/
  1255. int lp_add_service(char *pszService, int iDefaultService)
  1256. {
  1257.   return(add_a_service(pSERVICE(iDefaultService),pszService));
  1258. }
  1259.  
  1260.  
  1261. /***************************************************************************
  1262. add the IPC service
  1263. ***************************************************************************/
  1264. static BOOL lp_add_ipc(void)
  1265. {
  1266.   pstring comment;
  1267.   int i = add_a_service(&sDefault,"IPC$");
  1268.  
  1269.   if (i < 0)
  1270.     return(False);
  1271.  
  1272.   slprintf(comment,sizeof(comment),
  1273.        "IPC Service (%s)", Globals.szServerString );
  1274.  
  1275.   string_set(&iSERVICE(i).szPath,tmpdir());
  1276.   string_set(&iSERVICE(i).szUsername,"");
  1277.   string_set(&iSERVICE(i).comment,comment);
  1278.   iSERVICE(i).status = False;
  1279.   iSERVICE(i).iMaxConnections = 0;
  1280.   iSERVICE(i).bAvailable = True;
  1281.   iSERVICE(i).bRead_only = True;
  1282.   iSERVICE(i).bGuest_only = False;
  1283.   iSERVICE(i).bGuest_ok = True;
  1284.   iSERVICE(i).bPrint_ok = False;
  1285.   iSERVICE(i).bBrowseable = sDefault.bBrowseable;
  1286.  
  1287.   DEBUG(3,("adding IPC service\n"));
  1288.  
  1289.   return(True);
  1290. }
  1291.  
  1292.  
  1293. /***************************************************************************
  1294. add a new printer service, with defaults coming from service iFrom
  1295. ***************************************************************************/
  1296. BOOL lp_add_printer(char *pszPrintername, int iDefaultService)
  1297. {
  1298.   char *comment = "From Printcap";
  1299.   int i = add_a_service(pSERVICE(iDefaultService),pszPrintername);
  1300.   
  1301.   if (i < 0)
  1302.     return(False);
  1303.   
  1304.   /* note that we do NOT default the availability flag to True - */
  1305.   /* we take it from the default service passed. This allows all */
  1306.   /* dynamic printers to be disabled by disabling the [printers] */
  1307.   /* entry (if/when the 'available' keyword is implemented!).    */
  1308.   
  1309.   /* the printer name is set to the service name. */
  1310.   string_set(&iSERVICE(i).szPrintername,pszPrintername);
  1311.   string_set(&iSERVICE(i).comment,comment);
  1312.   iSERVICE(i).bBrowseable = sDefault.bBrowseable;
  1313.   /* Printers cannot be read_only. */
  1314.   iSERVICE(i).bRead_only = False;
  1315.   /* No share modes on printer services. */
  1316.   iSERVICE(i).bShareModes = False;
  1317.   /* No oplocks on printer services. */
  1318.   iSERVICE(i).bOpLocks = False;
  1319.   /* Printer services must be printable. */
  1320.   iSERVICE(i).bPrint_ok = True;
  1321.   
  1322.   DEBUG(3,("adding printer service %s\n",pszPrintername));
  1323.   
  1324.   return(True);
  1325. }
  1326.  
  1327.  
  1328. /***************************************************************************
  1329. Do a case-insensitive, whitespace-ignoring string compare.
  1330. ***************************************************************************/
  1331. static int strwicmp(char *psz1, char *psz2)
  1332. {
  1333.    /* if BOTH strings are NULL, return TRUE, if ONE is NULL return */
  1334.    /* appropriate value. */
  1335.    if (psz1 == psz2)
  1336.       return (0);
  1337.    else
  1338.       if (psz1 == NULL)
  1339.          return (-1);
  1340.       else
  1341.           if (psz2 == NULL)
  1342.               return (1);
  1343.  
  1344.    /* sync the strings on first non-whitespace */
  1345.    while (1)
  1346.    {
  1347.       while (isspace(*psz1))
  1348.          psz1++;
  1349.       while (isspace(*psz2))
  1350.          psz2++;
  1351.       if (toupper(*psz1) != toupper(*psz2) || *psz1 == '\0' || *psz2 == '\0')
  1352.          break;
  1353.       psz1++;
  1354.       psz2++;
  1355.    }
  1356.    return (*psz1 - *psz2);
  1357. }
  1358.  
  1359. /***************************************************************************
  1360. Map a parameter's string representation to something we can use. 
  1361. Returns False if the parameter string is not recognised, else TRUE.
  1362. ***************************************************************************/
  1363. static int map_parameter(char *pszParmName)
  1364. {
  1365.    int iIndex;
  1366.  
  1367.    if (*pszParmName == '-')
  1368.      return(-1);
  1369.  
  1370.    for (iIndex = 0; parm_table[iIndex].label; iIndex++) 
  1371.       if (strwicmp(parm_table[iIndex].label, pszParmName) == 0)
  1372.          return(iIndex);
  1373.  
  1374.    DEBUG(0,( "Unknown parameter encountered: \"%s\"\n", pszParmName));
  1375.    return(-1);
  1376. }
  1377.  
  1378.  
  1379. /***************************************************************************
  1380. Set a boolean variable from the text value stored in the passed string.
  1381. Returns True in success, False if the passed string does not correctly 
  1382. represent a boolean.
  1383. ***************************************************************************/
  1384. static BOOL set_boolean(BOOL *pb, char *pszParmValue)
  1385. {
  1386.    BOOL bRetval;
  1387.  
  1388.    bRetval = True;
  1389.    if (strwicmp(pszParmValue, "yes") == 0 ||
  1390.        strwicmp(pszParmValue, "true") == 0 ||
  1391.        strwicmp(pszParmValue, "1") == 0)
  1392.       *pb = True;
  1393.    else
  1394.       if (strwicmp(pszParmValue, "no") == 0 ||
  1395.           strwicmp(pszParmValue, "False") == 0 ||
  1396.           strwicmp(pszParmValue, "0") == 0)
  1397.          *pb = False;
  1398.       else
  1399.       {
  1400.          DEBUG(0,( "Badly formed boolean in configuration file: \"%s\".\n",
  1401.                pszParmValue));
  1402.          bRetval = False;
  1403.       }
  1404.    return (bRetval);
  1405. }
  1406.  
  1407. /***************************************************************************
  1408. Find a service by name. Otherwise works like get_service.
  1409. ***************************************************************************/
  1410. static int getservicebyname(char *pszServiceName, service *pserviceDest)
  1411. {
  1412.    int iService;
  1413.  
  1414.    for (iService = iNumServices - 1; iService >= 0; iService--)
  1415.       if (VALID(iService) &&
  1416.       strwicmp(iSERVICE(iService).szService, pszServiceName) == 0) 
  1417.       {
  1418.          if (pserviceDest != NULL)
  1419.        copy_service(pserviceDest, pSERVICE(iService), NULL);
  1420.          break;
  1421.       }
  1422.  
  1423.    return (iService);
  1424. }
  1425.  
  1426.  
  1427.  
  1428. /***************************************************************************
  1429. Copy a service structure to another
  1430.  
  1431. If pcopymapDest is NULL then copy all fields
  1432. ***************************************************************************/
  1433. static void copy_service(service *pserviceDest, 
  1434.                          service *pserviceSource,
  1435.                          BOOL *pcopymapDest)
  1436. {
  1437.   int i;
  1438.   BOOL bcopyall = (pcopymapDest == NULL);
  1439.  
  1440.   for (i=0;parm_table[i].label;i++)
  1441.     if (parm_table[i].ptr && parm_table[i].class == P_LOCAL && 
  1442.     (bcopyall || pcopymapDest[i]))
  1443.       {
  1444.     void *def_ptr = parm_table[i].ptr;
  1445.     void *src_ptr = 
  1446.       ((char *)pserviceSource) + PTR_DIFF(def_ptr,&sDefault);
  1447.     void *dest_ptr = 
  1448.       ((char *)pserviceDest) + PTR_DIFF(def_ptr,&sDefault);
  1449.  
  1450.     switch (parm_table[i].type)
  1451.       {
  1452.       case P_BOOL:
  1453.       case P_BOOLREV:
  1454.         *(BOOL *)dest_ptr = *(BOOL *)src_ptr;
  1455.         break;
  1456.  
  1457.       case P_INTEGER:
  1458.       case P_ENUM:
  1459.       case P_OCTAL:
  1460.         *(int *)dest_ptr = *(int *)src_ptr;
  1461.         break;
  1462.  
  1463.       case P_CHAR:
  1464.         *(char *)dest_ptr = *(char *)src_ptr;
  1465.         break;
  1466.  
  1467.       case P_STRING:
  1468.         string_set(dest_ptr,*(char **)src_ptr);
  1469.         break;
  1470.  
  1471.       case P_USTRING:
  1472.         string_set(dest_ptr,*(char **)src_ptr);
  1473.         strupper(*(char **)dest_ptr);
  1474.         break;
  1475.       default:
  1476.         break;
  1477.       }
  1478.       }
  1479.  
  1480.   if (bcopyall)
  1481.     {
  1482.       init_copymap(pserviceDest);
  1483.       if (pserviceSource->copymap)
  1484.     memcpy((void *)pserviceDest->copymap,
  1485.            (void *)pserviceSource->copymap,sizeof(BOOL)*NUMPARAMETERS);
  1486.     }
  1487. }
  1488.  
  1489. /***************************************************************************
  1490. Check a service for consistency. Return False if the service is in any way
  1491. incomplete or faulty, else True.
  1492. ***************************************************************************/
  1493. static BOOL service_ok(int iService)
  1494. {
  1495.    BOOL bRetval;
  1496.  
  1497.    bRetval = True;
  1498.    if (iSERVICE(iService).szService[0] == '\0')
  1499.    {
  1500.       DEBUG(0,( "The following message indicates an internal error:\n"));
  1501.       DEBUG(0,( "No service name in service entry.\n"));
  1502.       bRetval = False;
  1503.    }
  1504.  
  1505.    /* The [printers] entry MUST be printable. I'm all for flexibility, but */
  1506.    /* I can't see why you'd want a non-printable printer service...        */
  1507.    if (strwicmp(iSERVICE(iService).szService,PRINTERS_NAME) == 0)
  1508.       if (!iSERVICE(iService).bPrint_ok)
  1509.       {
  1510.          DEBUG(0,( "WARNING: [%s] service MUST be printable!\n",
  1511.                iSERVICE(iService).szService));
  1512.      iSERVICE(iService).bPrint_ok = True;
  1513.       }
  1514.  
  1515.    if (iSERVICE(iService).szPath[0] == '\0' &&
  1516.        strwicmp(iSERVICE(iService).szService,HOMES_NAME) != 0)
  1517.    {
  1518.       DEBUG(0,("No path in service %s - using %s\n",iSERVICE(iService).szService,tmpdir()));
  1519.       string_set(&iSERVICE(iService).szPath,tmpdir());      
  1520.    }
  1521.  
  1522.    /* If a service is flagged unavailable, log the fact at level 0. */
  1523.    if (!iSERVICE(iService).bAvailable) 
  1524.       DEBUG(1,( "NOTE: Service %s is flagged unavailable.\n",
  1525.             iSERVICE(iService).szService));
  1526.  
  1527.    return (bRetval);
  1528. }
  1529.  
  1530. static struct file_lists {
  1531.   struct file_lists *next;
  1532.   char *name;
  1533.   time_t modtime;
  1534. } *file_lists = NULL;
  1535.  
  1536. /*******************************************************************
  1537. keep a linked list of all config files so we know when one has changed 
  1538. it's date and needs to be reloaded
  1539. ********************************************************************/
  1540. static void add_to_file_list(char *fname)
  1541. {
  1542.   struct file_lists *f=file_lists;
  1543.  
  1544.   while (f) {
  1545.     if (f->name && !strcmp(f->name,fname)) break;
  1546.     f = f->next;
  1547.   }
  1548.  
  1549.   if (!f) {
  1550.     f = (struct file_lists *)malloc(sizeof(file_lists[0]));
  1551.     if (!f) return;
  1552.     f->next = file_lists;
  1553.     f->name = strdup(fname);
  1554.     if (!f->name) {
  1555.       free(f);
  1556.       return;
  1557.     }
  1558.     file_lists = f;
  1559.   }
  1560.  
  1561.   {
  1562.     pstring n2;
  1563.     pstrcpy(n2,fname);
  1564.     standard_sub_basic(n2);
  1565.     f->modtime = file_modtime(n2);
  1566.   }
  1567.  
  1568. }
  1569.  
  1570. /*******************************************************************
  1571. check if a config file has changed date
  1572. ********************************************************************/
  1573. BOOL lp_file_list_changed(void)
  1574. {
  1575.   struct file_lists *f = file_lists;
  1576.   DEBUG(6,("lp_file_list_changed()\n"));
  1577.  
  1578.   while (f)
  1579.   {
  1580.     pstring n2;
  1581.     time_t mod_time;
  1582.  
  1583.     pstrcpy(n2,f->name);
  1584.     standard_sub_basic(n2);
  1585.  
  1586.     DEBUG(6,("file %s -> %s  last mod_time: %s\n",
  1587.              f->name, n2, ctime(&f->modtime)));
  1588.  
  1589.     mod_time = file_modtime(n2);
  1590.  
  1591.     if (f->modtime != mod_time) {
  1592.         DEBUG(6,("file %s modified: %s\n", n2, ctime(&mod_time)));
  1593.         f->modtime = mod_time;
  1594.         return(True);
  1595.     }
  1596.     f = f->next;   
  1597.   }
  1598.   return(False);
  1599. }
  1600.  
  1601. /***************************************************************************
  1602.   handle the interpretation of the coding system parameter
  1603.   *************************************************************************/
  1604. static BOOL handle_coding_system(char *pszParmValue,char **ptr)
  1605. {
  1606.     string_set(ptr,pszParmValue);
  1607.     interpret_coding_system(pszParmValue);
  1608.     return(True);
  1609. }
  1610.  
  1611. /***************************************************************************
  1612. handle the interpretation of the character set system parameter
  1613. ***************************************************************************/
  1614. static BOOL handle_character_set(char *pszParmValue,char **ptr)
  1615. {
  1616.     string_set(ptr,pszParmValue);
  1617.     interpret_character_set(pszParmValue);
  1618.     return(True);
  1619. }
  1620.  
  1621.  
  1622. /***************************************************************************
  1623. handle the valid chars lines
  1624. ***************************************************************************/
  1625. static BOOL handle_valid_chars(char *pszParmValue,char **ptr)
  1626.   string_set(ptr,pszParmValue);
  1627.  
  1628.   /* A dependency here is that the parameter client code page must be
  1629.      set before this is called - as calling codepage_initialise()
  1630.      would overwrite the valid char lines.
  1631.    */
  1632.   codepage_initialise(lp_client_code_page());
  1633.  
  1634.   add_char_string(pszParmValue);
  1635.   return(True);
  1636. }
  1637.  
  1638.  
  1639. /***************************************************************************
  1640. handle the include operation
  1641. ***************************************************************************/
  1642. static BOOL handle_include(char *pszParmValue,char **ptr)
  1643.   pstring fname;
  1644.   pstrcpy(fname,pszParmValue);
  1645.  
  1646.   add_to_file_list(fname);
  1647.  
  1648.   standard_sub_basic(fname);
  1649.  
  1650.   string_set(ptr,fname);
  1651.  
  1652.   if (file_exist(fname,NULL))
  1653.     return(pm_process(fname, do_section, do_parameter));      
  1654.  
  1655.   DEBUG(2,("Can't find include file %s\n",fname));
  1656.  
  1657.   return(False);
  1658. }
  1659.  
  1660.  
  1661. /***************************************************************************
  1662. handle the interpretation of the copy parameter
  1663. ***************************************************************************/
  1664. static BOOL handle_copy(char *pszParmValue,char **ptr)
  1665. {
  1666.    BOOL bRetval;
  1667.    int iTemp;
  1668.    service serviceTemp;
  1669.  
  1670.    string_set(ptr,pszParmValue);
  1671.  
  1672.    init_service(&serviceTemp);
  1673.  
  1674.    bRetval = False;
  1675.    
  1676.    DEBUG(3,("Copying service from service %s\n",pszParmValue));
  1677.  
  1678.    if ((iTemp = getservicebyname(pszParmValue, &serviceTemp)) >= 0)
  1679.      {
  1680.        if (iTemp == iServiceIndex)
  1681.      {
  1682.        DEBUG(0,("Can't copy service %s - unable to copy self!\n",
  1683.             pszParmValue));
  1684.      }
  1685.        else
  1686.      {
  1687.        copy_service(pSERVICE(iServiceIndex), 
  1688.             &serviceTemp,
  1689.             iSERVICE(iServiceIndex).copymap);
  1690.        bRetval = True;
  1691.      }
  1692.      }
  1693.    else
  1694.      {
  1695.        DEBUG(0,( "Unable to copy service - source not found: %s\n",
  1696.         pszParmValue));
  1697.        bRetval = False;
  1698.      }
  1699.  
  1700.    free_service(&serviceTemp);
  1701.    return (bRetval);
  1702. }
  1703.  
  1704.  
  1705. /***************************************************************************
  1706. initialise a copymap
  1707. ***************************************************************************/
  1708. static void init_copymap(service *pservice)
  1709. {
  1710.   int i;
  1711.   if (pservice->copymap) free(pservice->copymap);
  1712.   pservice->copymap = (BOOL *)malloc(sizeof(BOOL)*NUMPARAMETERS);
  1713.   if (!pservice->copymap)
  1714.     DEBUG(0,("Couldn't allocate copymap!! (size %d)\n",NUMPARAMETERS));
  1715.  
  1716.   for (i=0;i<NUMPARAMETERS;i++)
  1717.     pservice->copymap[i] = True;
  1718. }
  1719.  
  1720.  
  1721. /***************************************************************************
  1722. Process a parameter for a particular service number. If snum < 0
  1723. then assume we are in the globals
  1724. ***************************************************************************/
  1725. BOOL lp_do_parameter(int snum, char *pszParmName, char *pszParmValue)
  1726. {
  1727.    int parmnum, i;
  1728.    void *parm_ptr=NULL; /* where we are going to store the result */
  1729.    void *def_ptr=NULL;
  1730.  
  1731.    parmnum = map_parameter(pszParmName);
  1732.  
  1733.    if (parmnum < 0)
  1734.      {
  1735.        DEBUG(0,( "Ignoring unknown parameter \"%s\"\n", pszParmName));
  1736.        return(True);
  1737.      }
  1738.  
  1739.    def_ptr = parm_table[parmnum].ptr;
  1740.  
  1741.    /* we might point at a service, the default service or a global */
  1742.    if (snum < 0) {
  1743.      parm_ptr = def_ptr;
  1744.    } else {
  1745.        if (parm_table[parmnum].class == P_GLOBAL) {
  1746.        DEBUG(0,( "Global parameter %s found in service section!\n",pszParmName));
  1747.        return(True);
  1748.      }
  1749.        parm_ptr = ((char *)pSERVICE(snum)) + PTR_DIFF(def_ptr,&sDefault);
  1750.    }
  1751.  
  1752.    if (snum >= 0) {
  1753.        if (!iSERVICE(snum).copymap)
  1754.            init_copymap(pSERVICE(snum));
  1755.        
  1756.        /* this handles the aliases - set the copymap for other entries with
  1757.           the same data pointer */
  1758.        for (i=0;parm_table[i].label;i++)
  1759.            if (parm_table[i].ptr == parm_table[parmnum].ptr)
  1760.                iSERVICE(snum).copymap[i] = False;
  1761.    }
  1762.  
  1763.    /* if it is a special case then go ahead */
  1764.    if (parm_table[parmnum].special) {
  1765.        parm_table[parmnum].special(pszParmValue,parm_ptr);
  1766.        return(True);
  1767.    }
  1768.  
  1769.    /* now switch on the type of variable it is */
  1770.    switch (parm_table[parmnum].type)
  1771.      {
  1772.      case P_BOOL:
  1773.        set_boolean(parm_ptr,pszParmValue);
  1774.        break;
  1775.  
  1776.      case P_BOOLREV:
  1777.        set_boolean(parm_ptr,pszParmValue);
  1778.        *(BOOL *)parm_ptr = ! *(BOOL *)parm_ptr;
  1779.        break;
  1780.  
  1781.      case P_INTEGER:
  1782.        *(int *)parm_ptr = atoi(pszParmValue);
  1783.        break;
  1784.  
  1785.      case P_CHAR:
  1786.        *(char *)parm_ptr = *pszParmValue;
  1787.        break;
  1788.  
  1789.      case P_OCTAL:
  1790.        sscanf(pszParmValue,"%o",(int *)parm_ptr);
  1791.        break;
  1792.  
  1793.      case P_STRING:
  1794.        string_set(parm_ptr,pszParmValue);
  1795.        break;
  1796.  
  1797.      case P_USTRING:
  1798.        string_set(parm_ptr,pszParmValue);
  1799.        strupper(*(char **)parm_ptr);
  1800.        break;
  1801.  
  1802.      case P_GSTRING:
  1803.        pstrcpy((char *)parm_ptr,pszParmValue);
  1804.        break;
  1805.  
  1806.      case P_UGSTRING:
  1807.        pstrcpy((char *)parm_ptr,pszParmValue);
  1808.        strupper((char *)parm_ptr);
  1809.        break;
  1810.  
  1811.      case P_ENUM:
  1812.          for (i=0;parm_table[parmnum].enum_list[i].name;i++) {
  1813.              if (strequal(pszParmValue, parm_table[parmnum].enum_list[i].name)) {
  1814.                  *(int *)parm_ptr = parm_table[parmnum].enum_list[i].value;
  1815.                  break;
  1816.              }
  1817.          }
  1818.          break;
  1819.      }
  1820.  
  1821.    return(True);
  1822. }
  1823.  
  1824. /***************************************************************************
  1825. Process a parameter.
  1826. ***************************************************************************/
  1827. static BOOL do_parameter(char *pszParmName, char *pszParmValue)
  1828. {
  1829.    if (!bInGlobalSection && bGlobalOnly) return(True);
  1830.  
  1831.    DEBUG(3,("doing parameter %s = %s\n",pszParmName,pszParmValue));
  1832.  
  1833.    return lp_do_parameter(bInGlobalSection?-2:iServiceIndex, pszParmName, pszParmValue);
  1834. }
  1835.  
  1836.  
  1837. /***************************************************************************
  1838. print a parameter of the specified type
  1839. ***************************************************************************/
  1840. static void print_parameter(struct parm_struct *p,void *ptr, FILE *f)
  1841. {
  1842.     int i;
  1843.     switch (p->type) {
  1844.     case P_ENUM:
  1845.         for (i=0;p->enum_list[i].name;i++) {
  1846.             if (*(int *)ptr == p->enum_list[i].value) {
  1847.                 fprintf(f,"%s",p->enum_list[i].name);
  1848.                 break;
  1849.             }
  1850.         }
  1851.         break;
  1852.  
  1853.     case P_BOOL:
  1854.         fprintf(f,"%s",BOOLSTR(*(BOOL *)ptr));
  1855.         break;
  1856.       
  1857.     case P_BOOLREV:
  1858.         fprintf(f,"%s",BOOLSTR(! *(BOOL *)ptr));
  1859.         break;
  1860.       
  1861.     case P_INTEGER:
  1862.         fprintf(f,"%d",*(int *)ptr);
  1863.         break;
  1864.       
  1865.     case P_CHAR:
  1866.         fprintf(f,"%c",*(char *)ptr);
  1867.         break;
  1868.       
  1869.     case P_OCTAL:
  1870.         fprintf(f,"0%o",*(int *)ptr);
  1871.         break;
  1872.       
  1873.     case P_GSTRING:
  1874.     case P_UGSTRING:
  1875.         if ((char *)ptr)
  1876.             fprintf(f,"%s",(char *)ptr);
  1877.         break;
  1878.         
  1879.     case P_STRING:
  1880.     case P_USTRING:
  1881.         if (*(char **)ptr)
  1882.             fprintf(f,"%s",*(char **)ptr);
  1883.         break;
  1884.     }
  1885. }
  1886.  
  1887.  
  1888. /***************************************************************************
  1889. print a parameter of the specified type
  1890. ***************************************************************************/
  1891. static void parameter_string(struct parm_struct *p,void *ptr,char *s)
  1892. {
  1893.     int i;
  1894.     s[0] = 0;
  1895.     
  1896.     switch (p->type) {
  1897.     case P_ENUM:
  1898.         for (i=0;p->enum_list[i].name;i++) {
  1899.             if (*(int *)ptr == p->enum_list[i].value) {
  1900.                 slprintf(s,sizeof(pstring)-1,"%s",p->enum_list[i].name);
  1901.                 break;
  1902.             }
  1903.         }
  1904.         break;
  1905.  
  1906.     case P_BOOL:
  1907.         slprintf(s, sizeof(pstring)-1, "%s",BOOLSTR(*(BOOL *)ptr));
  1908.         break;
  1909.         
  1910.     case P_BOOLREV:
  1911.         slprintf(s, sizeof(pstring)-1, "%s",BOOLSTR(! *(BOOL *)ptr));
  1912.         break;
  1913.         
  1914.     case P_INTEGER:
  1915.         slprintf(s, sizeof(pstring)-1, "%d",*(int *)ptr);
  1916.         break;
  1917.         
  1918.     case P_CHAR:
  1919.         slprintf(s, sizeof(pstring)-1, "%c",*(char *)ptr);
  1920.         break;
  1921.         
  1922.     case P_OCTAL:
  1923.         slprintf(s, sizeof(pstring)-1, "0%o",*(int *)ptr);
  1924.         break;
  1925.         
  1926.     case P_GSTRING:
  1927.     case P_UGSTRING:
  1928.         if ((char *)ptr)
  1929.             slprintf(s, sizeof(pstring)-1, "%s",(char *)ptr);
  1930.         break;
  1931.         
  1932.     case P_STRING:
  1933.     case P_USTRING:
  1934.         if (*(char **)ptr)
  1935.             slprintf(s, sizeof(pstring)-1,"%s",*(char **)ptr);
  1936.         break;
  1937.     }
  1938. }
  1939.  
  1940.  
  1941. /***************************************************************************
  1942. check if two parameters are equal
  1943. ***************************************************************************/
  1944. static BOOL equal_parameter(parm_type type,void *ptr1,void *ptr2)
  1945. {
  1946.   switch (type)
  1947.     {
  1948.     case P_BOOL:
  1949.     case P_BOOLREV:
  1950.       return(*((BOOL *)ptr1) == *((BOOL *)ptr2));
  1951.  
  1952.     case P_INTEGER:
  1953.     case P_ENUM:
  1954.     case P_OCTAL:
  1955.       return(*((int *)ptr1) == *((int *)ptr2));
  1956.       
  1957.     case P_CHAR:
  1958.       return(*((char *)ptr1) == *((char *)ptr2));
  1959.  
  1960.     case P_GSTRING:
  1961.     case P_UGSTRING:
  1962.       {
  1963.     char *p1 = (char *)ptr1, *p2 = (char *)ptr2;
  1964.     if (p1 && !*p1) p1 = NULL;
  1965.     if (p2 && !*p2) p2 = NULL;
  1966.     return(p1==p2 || strequal(p1,p2));
  1967.       }
  1968.     case P_STRING:
  1969.     case P_USTRING:
  1970.       {
  1971.     char *p1 = *(char **)ptr1, *p2 = *(char **)ptr2;
  1972.     if (p1 && !*p1) p1 = NULL;
  1973.     if (p2 && !*p2) p2 = NULL;
  1974.     return(p1==p2 || strequal(p1,p2));
  1975.       }
  1976.     }
  1977.   return(False);
  1978. }
  1979.  
  1980. /***************************************************************************
  1981. Process a new section (service). At this stage all sections are services.
  1982. Later we'll have special sections that permit server parameters to be set.
  1983. Returns True on success, False on failure.
  1984. ***************************************************************************/
  1985. static BOOL do_section(char *pszSectionName)
  1986. {
  1987.    BOOL bRetval;
  1988.    BOOL isglobal = ((strwicmp(pszSectionName, GLOBAL_NAME) == 0) || 
  1989.             (strwicmp(pszSectionName, GLOBAL_NAME2) == 0));
  1990.    bRetval = False;
  1991.  
  1992.    /* if we were in a global section then do the local inits */
  1993.    if (bInGlobalSection && !isglobal)
  1994.      init_locals();
  1995.  
  1996.    /* if we've just struck a global section, note the fact. */
  1997.    bInGlobalSection = isglobal;   
  1998.  
  1999.    /* check for multiple global sections */
  2000.    if (bInGlobalSection)
  2001.    {
  2002.      DEBUG(3,( "Processing section \"[%s]\"\n", pszSectionName));
  2003.      return(True);
  2004.    }
  2005.  
  2006.    if (!bInGlobalSection && bGlobalOnly) return(True);
  2007.  
  2008.    /* if we have a current service, tidy it up before moving on */
  2009.    bRetval = True;
  2010.  
  2011.    if (iServiceIndex >= 0)
  2012.      bRetval = service_ok(iServiceIndex);
  2013.  
  2014.    /* if all is still well, move to the next record in the services array */
  2015.    if (bRetval)
  2016.      {
  2017.        /* We put this here to avoid an odd message order if messages are */
  2018.        /* issued by the post-processing of a previous section. */
  2019.        DEBUG(2,( "Processing section \"[%s]\"\n", pszSectionName));
  2020.  
  2021.        if ((iServiceIndex=add_a_service(&sDefault,pszSectionName)) < 0)
  2022.      {
  2023.        DEBUG(0,("Failed to add a new service\n"));
  2024.        return(False);
  2025.      }
  2026.      }
  2027.  
  2028.    return (bRetval);
  2029. }
  2030.  
  2031. /***************************************************************************
  2032. Display the contents of the global structure.
  2033. ***************************************************************************/
  2034. static void dump_globals(FILE *f)
  2035. {
  2036.   int i;
  2037.   fprintf(f, "# Global parameters\n");
  2038.  
  2039.   for (i=0;parm_table[i].label;i++)
  2040.     if (parm_table[i].class == P_GLOBAL &&
  2041.     parm_table[i].ptr &&
  2042.     (i == 0 || (parm_table[i].ptr != parm_table[i-1].ptr)))
  2043.       {
  2044.     fprintf(f,"\t%s = ",parm_table[i].label);
  2045.     print_parameter(&parm_table[i],parm_table[i].ptr, f);
  2046.     fprintf(f,"\n");
  2047.       }
  2048. }
  2049.  
  2050. /***************************************************************************
  2051. Display the contents of a single services record.
  2052. ***************************************************************************/
  2053. static void dump_a_service(service *pService, FILE *f)
  2054. {
  2055.   int i;
  2056.   if (pService == &sDefault)
  2057.     fprintf(f,"\n\n# Default service parameters\n");
  2058.   else
  2059.     fprintf(f,"\n[%s]\n",pService->szService);
  2060.  
  2061.   for (i=0;parm_table[i].label;i++)
  2062.     if (parm_table[i].class == P_LOCAL &&
  2063.     parm_table[i].ptr && 
  2064.     (*parm_table[i].label != '-') &&
  2065.     (i == 0 || (parm_table[i].ptr != parm_table[i-1].ptr)))
  2066.       {
  2067.     int pdiff = PTR_DIFF(parm_table[i].ptr,&sDefault);
  2068.  
  2069.     if (pService == &sDefault || !equal_parameter(parm_table[i].type,
  2070.                               ((char *)pService) + pdiff,
  2071.                               ((char *)&sDefault) + pdiff))
  2072.       {
  2073.         fprintf(f,"\t%s = ",parm_table[i].label);
  2074.         print_parameter(&parm_table[i],
  2075.                 ((char *)pService) + pdiff, f);
  2076.         fprintf(f,"\n");
  2077.       }
  2078.       }
  2079. }
  2080.  
  2081.  
  2082. /***************************************************************************
  2083. return info about the next service  in a service. snum==-1 gives the default
  2084. serice and snum==-2 gives the globals
  2085.  
  2086. return 0 when out of parameters
  2087. ***************************************************************************/
  2088. int lp_next_parameter(int snum, int *i, char *label, 
  2089.                char *value, int allparameters)
  2090. {
  2091.     if (snum == -2) {
  2092.         /* do the globals */
  2093.         for (;parm_table[*i].label;(*i)++)
  2094.             if (parm_table[*i].class == P_GLOBAL &&
  2095.                 parm_table[*i].ptr && 
  2096.                 (*parm_table[*i].label != '-') &&
  2097.                 ((*i) == 0 || 
  2098.                  (parm_table[*i].ptr != parm_table[(*i)-1].ptr))) {
  2099.                 pstrcpy(label, parm_table[*i].label);
  2100.                 parameter_string(&parm_table[*i],
  2101.                          parm_table[*i].ptr,
  2102.                          value);
  2103.                 (*i)++;
  2104.                 return 1;
  2105.             }
  2106.         return 0;
  2107.     } else {
  2108.         service *pService = (snum==-1?&sDefault:pSERVICE(snum));
  2109.  
  2110.         for (;parm_table[*i].label;(*i)++)
  2111.             if (parm_table[*i].class == P_LOCAL &&
  2112.                 parm_table[*i].ptr && 
  2113.                 (*parm_table[*i].label != '-') &&
  2114.                 ((*i) == 0 || 
  2115.                  (parm_table[*i].ptr != parm_table[(*i)-1].ptr))) {
  2116.                 int pdiff = PTR_DIFF(parm_table[*i].ptr,&sDefault);
  2117.                 
  2118.                 if (snum == -1 || allparameters ||
  2119.                     !equal_parameter(parm_table[*i].type,
  2120.                              ((char *)pService) + pdiff,
  2121.                              ((char *)&sDefault) + pdiff)) {
  2122.                     pstrcpy(label, parm_table[*i].label);
  2123.                     parameter_string(&parm_table[*i],
  2124.                              ((char *)pService) + pdiff,
  2125.                              value);
  2126.                     (*i)++;
  2127.                     return 1;
  2128.                 }
  2129.             }
  2130.     }
  2131.  
  2132.   return 0;
  2133. }
  2134.  
  2135.  
  2136. #if 0
  2137. /***************************************************************************
  2138. Display the contents of a single copy structure.
  2139. ***************************************************************************/
  2140. static void dump_copy_map(BOOL *pcopymap)
  2141. {
  2142.   int i;
  2143.   if (!pcopymap) return;
  2144.  
  2145.   printf("\n\tNon-Copied parameters:\n");
  2146.  
  2147.   for (i=0;parm_table[i].label;i++)
  2148.     if (parm_table[i].class == P_LOCAL &&
  2149.     parm_table[i].ptr && !pcopymap[i] &&
  2150.     (i == 0 || (parm_table[i].ptr != parm_table[i-1].ptr)))
  2151.       {
  2152.     printf("\t\t%s\n",parm_table[i].label);
  2153.       }
  2154. }
  2155. #endif
  2156.  
  2157. /***************************************************************************
  2158. Return TRUE if the passed service number is within range.
  2159. ***************************************************************************/
  2160. BOOL lp_snum_ok(int iService)
  2161. {
  2162.    return (LP_SNUM_OK(iService) && iSERVICE(iService).bAvailable);
  2163. }
  2164.  
  2165.  
  2166. /***************************************************************************
  2167. auto-load some homes and printer services
  2168. ***************************************************************************/
  2169. static void lp_add_auto_services(char *str)
  2170. {
  2171.   char *s;
  2172.   char *p;
  2173.   int homes, printers;
  2174.  
  2175.   if (!str)
  2176.     return;
  2177.  
  2178.   s = strdup(str);
  2179.   if (!s) return;
  2180.  
  2181.   homes = lp_servicenumber(HOMES_NAME);    
  2182.   printers = lp_servicenumber(PRINTERS_NAME);
  2183.  
  2184.   for (p=strtok(s,LIST_SEP);p;p=strtok(NULL,LIST_SEP))
  2185.     {
  2186.       char *home = get_home_dir(p);
  2187.  
  2188.       if (lp_servicenumber(p) >= 0) continue;
  2189.  
  2190.       if (home && homes >= 0)
  2191.     {
  2192.       lp_add_home(p,homes,home);
  2193.       continue;
  2194.     }
  2195.  
  2196.       if (printers >= 0 && pcap_printername_ok(p,NULL))
  2197.     lp_add_printer(p,printers);
  2198.     }
  2199.   free(s);
  2200. }
  2201.  
  2202. /***************************************************************************
  2203. auto-load one printer
  2204. ***************************************************************************/
  2205. static void lp_add_one_printer(char *name,char *comment)
  2206. {
  2207.   int printers = lp_servicenumber(PRINTERS_NAME);
  2208.   int i;
  2209.  
  2210.   if (lp_servicenumber(name) < 0)
  2211.     {
  2212.       lp_add_printer(name,printers);
  2213.       if ((i=lp_servicenumber(name)) >= 0)
  2214.     string_set(&iSERVICE(i).comment,comment);
  2215.     }      
  2216. }
  2217.  
  2218.  
  2219. /***************************************************************************
  2220. auto-load printer services
  2221. ***************************************************************************/
  2222. static void lp_add_all_printers(void)
  2223. {
  2224.   int printers = lp_servicenumber(PRINTERS_NAME);
  2225.  
  2226.   if (printers < 0) return;
  2227.  
  2228.   pcap_printer_fn(lp_add_one_printer);
  2229. }
  2230.  
  2231. /***************************************************************************
  2232. have we loaded a services file yet?
  2233. ***************************************************************************/
  2234. BOOL lp_loaded(void)
  2235. {
  2236.   return(bLoaded);
  2237. }
  2238.  
  2239. /***************************************************************************
  2240. unload unused services
  2241. ***************************************************************************/
  2242. void lp_killunused(BOOL (*snumused)(int ))
  2243. {
  2244.   int i;
  2245.   for (i=0;i<iNumServices;i++)
  2246.     if (VALID(i) && (!snumused || !snumused(i)))
  2247.       {
  2248.     iSERVICE(i).valid = False;
  2249.     free_service(pSERVICE(i));
  2250.       }
  2251. }
  2252.  
  2253. /***************************************************************************
  2254. Load the services array from the services file. Return True on success, 
  2255. False on failure.
  2256. ***************************************************************************/
  2257. BOOL lp_load(char *pszFname,BOOL global_only)
  2258. {
  2259.   pstring n2;
  2260.   BOOL bRetval;
  2261.  
  2262.   add_to_file_list(pszFname);
  2263.  
  2264.   bRetval = False;
  2265.  
  2266.   bInGlobalSection = True;
  2267.   bGlobalOnly = global_only;
  2268.   
  2269.   init_globals();
  2270.   
  2271.   pstrcpy(n2,pszFname);
  2272.   standard_sub_basic(n2);
  2273.  
  2274.   /* We get sections first, so have to start 'behind' to make up */
  2275.   iServiceIndex = -1;
  2276.   bRetval = pm_process(n2, do_section, do_parameter);
  2277.   
  2278.   /* finish up the last section */
  2279.   DEBUG(3,("pm_process() returned %s\n", BOOLSTR(bRetval)));
  2280.   if (bRetval)
  2281.     if (iServiceIndex >= 0)
  2282.       bRetval = service_ok(iServiceIndex);       
  2283.  
  2284.   lp_add_auto_services(lp_auto_services());
  2285.   if (lp_load_printers())
  2286.     lp_add_all_printers();
  2287.  
  2288.   lp_add_ipc();
  2289.  
  2290.   set_default_server_announce_type();
  2291.  
  2292.   bLoaded = True;
  2293.  
  2294.   return (bRetval);
  2295. }
  2296.  
  2297.  
  2298. /***************************************************************************
  2299. return the max number of services
  2300. ***************************************************************************/
  2301. int lp_numservices(void)
  2302. {
  2303.   return(iNumServices);
  2304. }
  2305.  
  2306. /***************************************************************************
  2307. Display the contents of the services array in human-readable form.
  2308. ***************************************************************************/
  2309. void lp_dump(FILE *f)
  2310. {
  2311.    int iService;
  2312.  
  2313.    dump_globals(f);
  2314.    
  2315.    dump_a_service(&sDefault, f);
  2316.  
  2317.    for (iService = 0; iService < iNumServices; iService++)
  2318.    {
  2319.      if (VALID(iService))
  2320.        {
  2321.      if (iSERVICE(iService).szService[0] == '\0')
  2322.        break;
  2323.      dump_a_service(pSERVICE(iService), f);
  2324.        }
  2325.    }
  2326. }
  2327.  
  2328.  
  2329. /***************************************************************************
  2330. Return the number of the service with the given name, or -1 if it doesn't
  2331. exist. Note that this is a DIFFERENT ANIMAL from the internal function
  2332. getservicebyname()! This works ONLY if all services have been loaded, and
  2333. does not copy the found service.
  2334. ***************************************************************************/
  2335. int lp_servicenumber(char *pszServiceName)
  2336. {
  2337.    int iService;
  2338.  
  2339.    for (iService = iNumServices - 1; iService >= 0; iService--)
  2340.       if (VALID(iService) &&
  2341.       strequal(lp_servicename(iService), pszServiceName)) 
  2342.          break;
  2343.  
  2344.    if (iService < 0)
  2345.      DEBUG(7,("lp_servicenumber: couldn't find %s\n",pszServiceName));
  2346.    
  2347.    return (iService);
  2348. }
  2349.  
  2350. /*******************************************************************
  2351.   a useful volume label function
  2352.   ******************************************************************/
  2353. char *volume_label(int snum)
  2354. {
  2355.   char *ret = lp_volume(snum);
  2356.   if (!*ret) return(lp_servicename(snum));
  2357.   return(ret);
  2358. }
  2359.  
  2360. #if 0
  2361. /*
  2362.  * nmbd only loads the global section. There seems to be no way to
  2363.  * determine exactly is a service is printable by only looking at the
  2364.  * [global] section so for now always announce as a print server. This
  2365.  * will need looking at in the future. Jeremy (jallison@whistle.com).
  2366.  */
  2367. /*******************************************************************
  2368.  Return true if any printer services are defined.
  2369.   ******************************************************************/
  2370. static BOOL lp_printer_services(void)
  2371. {
  2372.   int iService;
  2373.  
  2374.   for (iService = iNumServices - 1; iService >= 0; iService--)
  2375.       if (VALID(iService) && iSERVICE(iService).bPrint_ok)
  2376.           return True;
  2377.   return False;
  2378. }
  2379. #endif
  2380.  
  2381. /*******************************************************************
  2382.  Set the server type we will announce as via nmbd.
  2383. ********************************************************************/
  2384. static void set_default_server_announce_type()
  2385. {
  2386.   default_server_announce = (SV_TYPE_WORKSTATION | SV_TYPE_SERVER |
  2387.                               SV_TYPE_SERVER_UNIX | SV_TYPE_PRINTQ_SERVER);
  2388.   if(lp_announce_as() == ANNOUNCE_AS_NT)
  2389.     default_server_announce |= (SV_TYPE_SERVER_NT | SV_TYPE_NT);
  2390.   else if(lp_announce_as() == ANNOUNCE_AS_WIN95)
  2391.     default_server_announce |= SV_TYPE_WIN95_PLUS;
  2392.   else if(lp_announce_as() == ANNOUNCE_AS_WFW)
  2393.     default_server_announce |= SV_TYPE_WFW;
  2394.   default_server_announce |= (lp_time_server() ? SV_TYPE_TIME_SOURCE : 0);
  2395. /*
  2396.  * nmbd only loads the [global] section. There seems to be no way to
  2397.  * determine exactly if any service is printable by only looking at the
  2398.  * [global] section so for now always announce as a print server. This
  2399.  * will need looking at in the future. Jeremy (jallison@whistle.com).
  2400.  */
  2401. #if 0
  2402.   default_server_announce |= (lp_printer_services() ? SV_TYPE_PRINTQ_SERVER : 0);
  2403. #endif
  2404. }
  2405.  
  2406.  
  2407. /*******************************************************************
  2408. rename a service
  2409. ********************************************************************/
  2410. void lp_rename_service(int snum, char *new_name)
  2411. {
  2412.     string_set(&pSERVICE(snum)->szService, new_name);
  2413. }
  2414.  
  2415. /*******************************************************************
  2416. remove a service
  2417. ********************************************************************/
  2418. void lp_remove_service(int snum)
  2419. {
  2420.     pSERVICE(snum)->valid = False;
  2421. }
  2422.  
  2423. /*******************************************************************
  2424. copy a service
  2425. ********************************************************************/
  2426. void lp_copy_service(int snum, char *new_name)
  2427. {
  2428.     char *oldname = lp_servicename(snum);
  2429.     do_section(new_name);
  2430.     if (snum >= 0) {
  2431.         snum = lp_servicenumber(new_name);
  2432.         if (snum >= 0)
  2433.             lp_do_parameter(snum, "copy", oldname);
  2434.     }
  2435. }
  2436.  
  2437.  
  2438. /*******************************************************************
  2439.  Get the default server type we will announce as via nmbd.
  2440. ********************************************************************/
  2441. int lp_default_server_announce(void)
  2442. {
  2443.   return default_server_announce;
  2444. }
  2445.  
  2446. /*******************************************************************
  2447.  Split the announce version into major and minor numbers.
  2448. ********************************************************************/
  2449. int lp_major_announce_version(void)
  2450. {
  2451.   static BOOL got_major = False;
  2452.   static int major_version = DEFAULT_MAJOR_VERSION;
  2453.   char *vers;
  2454.   char *p;
  2455.  
  2456.   if(got_major)
  2457.     return major_version;
  2458.  
  2459.   got_major = True;
  2460.   if((vers = lp_announce_version()) == NULL)
  2461.     return major_version;
  2462.   
  2463.   if((p = strchr(vers, '.')) == 0)
  2464.     return major_version;
  2465.  
  2466.   *p = '\0';
  2467.   major_version = atoi(vers);
  2468.   return major_version;
  2469. }
  2470.  
  2471. int lp_minor_announce_version(void)
  2472. {
  2473.   static BOOL got_minor = False;
  2474.   static int minor_version = DEFAULT_MINOR_VERSION;
  2475.   char *vers;
  2476.   char *p;
  2477.  
  2478.   if(got_minor)
  2479.     return minor_version;
  2480.  
  2481.   got_minor = True;
  2482.   if((vers = lp_announce_version()) == NULL)
  2483.     return minor_version;
  2484.   
  2485.   if((p = strchr(vers, '.')) == 0)              
  2486.     return minor_version;
  2487.     
  2488.   p++;
  2489.   minor_version = atoi(p);
  2490.   return minor_version;
  2491. }  
  2492.  
  2493. /***********************************************************
  2494.  Set the global name resolution order (used in smbclient).
  2495. ************************************************************/
  2496.  
  2497. void lp_set_name_resolve_order(char *new_order)
  2498. {
  2499.   Globals.szNameResolveOrder = new_order;
  2500. }
  2501.